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PREFACE 


T his book is intended to survey the most important computer algorithms in use today, 
and to teach fundamental techniques to the growing number of people in need of 
knowing them. It is intended for use as a textbook for a second course in computer 
science, after students have acquired basic programming skills and familiarity with computer 
systems. The book also may be useful for self-study or as a reference for people engaged in 
the development of computer systems or applications programs, since it contains implemen¬ 
tations of useful algorithms and detailed information on performance characteristics and 
clients. The broad perspective taken makes the book an appropriate introduction to the field. 

the study of algorithms and data structures is fundamental to any computer- 
science curriculum, but it is not just for programmers and computer-science students. Every¬ 
one who uses a computer wants it to run faster or to solve larger problems. The algorithms 
in this book represent a body of knowledge developed over the last 50 years that has become 
indispensable. From /V-body simulation problems in physics to genetic-sequencing problems 
in molecular biology, the basic methods described here have become essential in scientific 
research; from architectural modeling systems to aircraft simulation, they have become es¬ 
sential tools in engineering; and from database systems to internet search engines, they have 
become essential parts of modern software systems. And these are but a few examples—as the 
scope of computer applications continues to grow, so grows the impact of the basic methods 
covered here. 

Before developing our fundamental approach to studying algorithms, we develop data 
types for stacks, queues, and other low-level abstractions that we use throughout the book. 
Then we survey fundamental algorithms for sorting, searching, graphs, and strings. The last 
chapter is an overview placing the rest of the material in the book in a larger context. 


viii 
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Distinctive features The orientation of the book is to study algorithms likely to be of 
practical use. The book teaches a broad variety of algorithms and data structures and pro¬ 
vides sufficient information about them that readers can confidently implement, debug, and 
put them to work in any computational environment. The approach involves: 

Algorithms. Our descriptions of algorithms are based on complete implementations and on 
a discussion of the operations of these programs on a consistent set of examples. Instead of 
presenting pseudo-code, we work with real code, so that the programs can quickly be put to 
practical use. Our programs are written in Java, but in a style such that most of our code can 
be reused to develop implementations in other modern programming languages. 

Data types. We use a modern programming style based on data abstraction, so that algo¬ 
rithms and their data structures are encapsulated together. 

Applications. Each chapter has a detailed description of applications where the algorithms 
described play a critical role. These range from applications in physics and molecular biology, 
to engineering computers and systems, to familiar tasks such as data compression and search¬ 
ing on the web. 

A scientific approach. We emphasize developing mathematical models for describing the 
performance of algorithms, using the models to develop hypotheses about performance, and 
then testing the hypotheses by running the algorithms in realistic contexts. 

Breadth of coverage. We cover basic abstract data types, sorting algorithms, searching al¬ 
gorithms, graph processing, and string processing. We keep the material in algorithmic con¬ 
text, describing data structures, algorithm design paradigms, reduction, and problem-solving 
models. We cover classic methods that have been taught since the 1960s and new methods 
that have been invented in recent years. 

Our primary goal is to introduce the most important algorithms in use today to as wide an 
audience as possible. These algorithms are generally ingenious creations that, remarkably, can 
each be expressed in just a dozen or two lines of code. As a group, they represent problem¬ 
solving power of amazing scope. They have enabled the construction of computational ar¬ 
tifacts, the solution of scientific problems, and the development of commercial applications 
that would not have been feasible without them. 
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Booksite An important feature of the book is its relationship to the booksite 
al gs4. cs. pri nceton. edu. This site is freely available and contains an extensive amount of 
material about algorithms and data structures, for teachers, students, and practitioners, in¬ 
cluding: 

An online synopsis. The text is summarized in the booksite to give it the same overall struc¬ 
ture as the book, but linked so as to provide easy navigation through the material. 

Full implementations. All code in the book is available on the booksite, in a form suitable for 
program development. Many other implementations are also available, including advanced 
implementations and improvements described in the book, answers to selected exercises, and 
client code for various applications. The emphasis is on testing algorithms in the context of 
meaningful applications. 

Exercises and answers. The booksite expands on the exercises in the book by adding drill 
exercises (with answers available with a click), a wide variety of examples illustrating the 
reach of the material, programming exercises with code solutions, and challenging problems. 

Dynamic visualizations. Dynamic simulations are impossible in a printed book, but the 
website is replete with implementations that use a graphics class to present compelling visual 
demonstrations of algorithm applications. 

Course materials. A complete set of lecture slides is tied directly to the material in the book 
and on the booksite. A full selection of programming assignments, with check lists, test data, 
and preparatory material, is also included. 

Links to related material. Hundreds of links lead students to background information about 
applications and to resources for studying algorithms. 

Our goal in creating this material was to provide a complementary approach to the ideas. 
Generally, you should read the book when learning specific algorithms for the first time or 
when trying to get a global picture, and you should use the booksite as a reference when pro¬ 
gramming or as a starting point when searching for more detail while online. 
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Use in the curriculum The book is intended as a textbook in a second course in com¬ 
puter science. It provides full coverage of core material and is an excellent vehicle for stu¬ 
dents to gain experience and maturity in programming, quantitative reasoning, and problem¬ 
solving. Typically, one course in computer science will suffice as a prerequisite—the book is 
intended for anyone conversant with a modern programming language and with the basic 
features of modern computer systems. 

The algorithms and data structures are expressed in lava, but in a style accessible to 
people fluent in other modern languages. We embrace modern Java abstractions (including 
generics) but resist dependence upon esoteric features of the language. 

Most of the mathematical material supporting the analytic results is self-contained (or 
is labeled as beyond the scope of this book), so little specific preparation in mathematics is 
required for the bulk of the book, although mathematical maturity is definitely helpful. Ap¬ 
plications are drawn from introductory material in the sciences, again self-contained. 

The material covered is a fundamental background for any student intending to major 
in computer science, electrical engineering, or operations research, and is valuable for any 
student with interests in science, mathematics, or engineering. 

Context The book is intended to follow our introductory text, An Introduction to Pro¬ 
gramming in Java: An Interdisciplinary Approach, which is a broad introduction to the field. 
Together, these two books can support a two- or three-semester introduction to computer sci¬ 
ence that will give any student the requisite background to successfully address computation 
in any chosen field of study in science, engineering, or the social sciences. 

The starting point for much of the material in the book was the Sedgewick series of Al¬ 
gorithms books. In spirit, this book is closest to the first and second editions of that book, but 
this text benefits from decades of experience teaching and learning that material. Sedgewick’s 
current Algorithms in C/C++/Java, Third Edition is more appropriate as a reference or a text 
for an advanced course; this book is specifically designed to be a textbook for a one-semester 
course for first- or second-year college students and as a modern introduction to the basics 
and a reference for use by working programmers. 
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T he objective of this book is to study a broad variety of important and useful 
algorithms —methods for solving problems that are suited for computer imple¬ 
mentation. Algorithms go hand in hand with data structures —schemes for or¬ 
ganizing data that leave them amenable to efficient processing by an algorithm. This 
chapter introduces the basic tools that we need to study algorithms and data structures. 

First, we introduce our basic programming model. All of our programs are imple¬ 
mented using a small subset of the Java programming language plus a few of our own 
libraries for input/output and for statistical calculations, section 1.1 is a summary of 
language constructs, features, and libraries that we use in this book. 

Next, we emphasize data abstraction, where we define abstract data types (ADTs) in 
the service of modular programming. In section 1.2 we introduce the process of im¬ 
plementing an ADT in Java, by specifying an applications programming interface (API) 
and then using the Java class mechanism to develop an implementation for use in client 
code. 

As important and useful examples, we next consider three fundamental ADTs: the 
bag, the queue, and the stack, section 1.3 describes APIs and implementations of bags, 
queues, and stacks using arrays, resizing arrays, and linked lists that serve as models and 
starting points for algorithm implementations throughout the book. 

Performance is a central consideration in the study of algorithms, section 1.4 de¬ 
scribes our approach to analyzing algorithm performance. The basis of our approach is 
the scientific method: we develop hypotheses about performance, create mathematical 
models, and run experiments to test them, repeating the process as necessary. 

We conclude with a case study where we consider solutions to a connectivity problem 
that uses algorithms and data structures that implement the classic union-find ADT. 
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CHAPTER 1 ■ Fundamentals 


Algorithms When we write a computer program, we are generally implementing a 
method that has been devised previously to solve some problem. This method is often 
independent of the particular programming language being used—it is likely to be 
equally appropriate for many computers and many programming languages. It is the 
method, rather than the computer program itself, that specifies the steps that we can 
take to solve the problem. The term algorithm is used in computer science to describe 
a finite, deterministic, and effective problem-solving method suitable for implementa¬ 
tion as a computer program. Algorithms are the stuff of computer science: they are 
central objects of study in the field. 

We can define an algorithm by describing a procedure for solving a problem in a 
natural language, or by writing a computer program that implements the procedure, 
as shown at right for Euclid’s algorithm for finding the greatest common divisor of 
two numbers, a variant of which was devised 
over 2,300 years ago. If you are not familiar 
with Euclid’s algorithm, you are encour¬ 
aged to work exercise 1 . 1.24 and exercise 
1 . 1 . 25 , perhaps after reading section 1 . 1 . In 
this book, we use computer programs to de¬ 
scribe algorithms. One important reason for 
doing so is that it makes easier the task of 
checking whether they are finite, determin¬ 
istic, and effective, as required. But it is also 
important to recognize that a program in a 
particular language is just one way to express 
an algorithm. The fact that many of the al¬ 
gorithms in this book have been expressed 
in multiple programming languages over the 
past several decades reinforces the idea that each algorithm is a method suitable for 
implementation on any computer in any programming language. 

Most algorithms of interest involve organizing the data involved in the computa¬ 
tion. Such organization leads to data structures, which also are central objects of study 
in computer science. Algorithms and data structures go hand in hand. In this book we 
take the view that data structures exist as the byproducts or end products of algorithms 
and that we must therefore study them in order to understand the algorithms. Simple 
algorithms can give rise to complicated data structures and, conversely, complicated 
algorithms can use simple data structures. We shall study the properties of many data 
structures in this book; indeed, we might well have titled the book Algorithms and Data 
Structures. 


English-language description 

Compute the greatest common divisor of 
two nonnegative integers p and q as follows: 
If q is 0, the answer is p. If not, divide p by q 
and take the remainder r. The answer is the 
greatest common divisor of q and r. 

Java-language description 

public static int gcd(int p, int q) 

{ 

if (q == 0) return p; 
i nt r = p % q; 
return gcd(q, r); 

} 

Euclid's algorithm 
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When we use a computer to help us solve a problem, we typically are faced with a 
number of possible approaches. For small problems, it hardly matters which approach 
we use, as long as we have one that correctly solves the problem. For huge problems (or 
applications where we need to solve huge numbers of small problems), however, we 
quickly become motivated to devise methods that use time and space efficiently. 

The primary reason to learn about algorithms is that this discipline gives us the 
potential to reap huge savings, even to the point of enabling us to do tasks that would 
otherwise be impossible. In an application where we are processing millions of objects, 
it is not unusual to be able to make a program millions of times faster by using a well- 
designed algorithm. We shall see such examples on numerous occasions throughout 
the book. By contrast, investing additional money or time to buy and install a new 
computer holds the potential for speeding up a program by perhaps a factor of only 10 
or 100. Careful algorithm design is an extremely effective part of the process of solving 
a huge problem, whatever the applications area. 

When developing a huge or complex computer program, a great deal of effort must 
go into understanding and defining the problem to be solved, managing its complex¬ 
ity, and decomposing it into smaller subtasks that can be implemented easily. Often, 
many of the algorithms required after the decomposition are trivial to implement. In 
most cases, however, there are a few algorithms whose choice is critical because most 
of the system resources will be spent running those algorithms. These are the types of 
algorithms on which we concentrate in this book. We study fundamental algorithms 
that are useful for solving challenging problems in a broad variety of applications areas. 

The sharing of programs in computer systems is becoming more widespread, so 
although we might expect to be using a large fraction of the algorithms in this book, we 
also might expect to have to implement only a small fraction of them. For example, the 
Java libraries contain implementations of a host of fundamental algorithms. However, 
implementing simple versions of basic algorithms helps us to understand them bet¬ 
ter and thus to more effectively use and tune advanced versions from a library. More 
important, the opportunity to reimplement basic algorithms arises frequently. The pri¬ 
mary reason to do so is that we are faced, all too often, with completely new computing 
environments (hardware and software) with new features that old implementations 
may not use to best advantage. In this book, we concentrate on the simplest reasonable 
implementations of the best algorithms. We do pay careful attention to coding the criti¬ 
cal parts of the algorithms, and take pains to note where low-level optimization effort 
could be most beneficial. 

The choice of the best algorithm for a particular task can be a complicated process, 
perhaps involving sophisticated mathematical analysis. The branch of computer sci¬ 
ence that comprises the study of such questions is called analysis of algorithms. Many 
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of the algorithms that we study have been shown through analysis to have excellent 
theoretical performance; others are simply known to work well through experience. 
Our primary goal is to learn reasonable algorithms for important tasks, yet we shall also 
pay careful attention to comparative performance of the methods. We should not use 
an algorithm without having an idea of what resources it might consume, so we strive 
to be aware of how our algorithms might be expected to perform. 

Summary of topics As an overview, we describe the major parts of the book, giv¬ 
ing specific topics covered and an indication of our general orientation toward the 
material. This set of topics is intended to touch on as many fundamental algorithms as 
possible. Some of the areas covered are core computer-science areas that we study in 
depth to learn basic algorithms of wide applicability. Other algorithms that we discuss 
are from advanced fields of study within computer science and related fields. The algo¬ 
rithms that we consider are the products of decades of research and development and 
continue to play an essential role in the ever-expanding applications of computation. 

Fundamentals (chapter i ) in the context of this book are the basic principles and 
methodology that we use to implement, analyze, and compare algorithms. We consider 
our Java programming model, data abstraction, basic data structures, abstract data 
types for collections, methods of analyzing algorithm performance, and a case study. 

Sorting algorithms (chapter 2 ) for rearranging arrays in order are of fundamental 
importance. We consider a variety of algorithms in considerable depth, including in¬ 
sertion sort, selection sort, shellsort, quicksort, mergesort, and heapsort. We also en¬ 
counter algorithms for several related problems, including priority queues, selection, 
and merging. Many of these algorithms will find application as the basis for other algo¬ 
rithms later in the book. 

Searching algorithms (chapter 3 ) for finding specific items among large collections 
of items are also of fundamental importance. We discuss basic and advanced methods 
for searching, including binary search trees, balanced search trees, and hashing. We 
note relationships among these methods and compare performance. 

Graphs (chapter 4 ) are sets of objects and connections, possibly with weights and 
orientation. Graphs are useful models for a vast number of difficult and important 
problems, and the design of algorithms for processing graphs is a major field of study. 
We consider depth-first search, breadth-first search, connectivity problems, and sev¬ 
eral algorithms and applications, including Kruskal’s and Prim’s algorithms for finding 
minimum spanning tree and Dijkstra’s and the Bellman-Ford algorithms for solving 
shortest-paths problems. 
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Strings (chapter 5 ) are an essential data type in modern computing applications. 
We consider a range of methods for processing sequences of characters. We begin with 
faster algorithms for sorting and searching when keys are strings. Then we consider 
substring search, regular expression pattern matching, and data-compression algo¬ 
rithms. Again, an introduction to advanced topics is given through treatment of some 
elementary problems that are important in their own right. 

Context (chapter 6 ) helps us relate the material in the book to several other advanced 
fields of study, including scientific computing, operations research, and the theory of 
computing. We survey event-based simulation, B-trees, suffix arrays, maximum flow, 
and other advanced topics from an introductory viewpoint to develop appreciation for 
the interesting advanced fields of study where algorithms play a critical role. Finally, we 
describe search problems, reduction, and NP-completeness to introduce the theoretical 
underpinnings of the study of algorithms and relationships to material in this book. 


THE STUDY OF ALGORITHMS IS INTERESTING AND EXCITING because it is a new field 
(almost all the algorithms that we study are less than 50 years old, and some were just 
recently discovered) with a rich tradition (a few algorithms have been known for hun¬ 
dreds of years). New discoveries are constantly being made, but few algorithms are 
completely understood. In this book we shall consider intricate, complicated, and diffi¬ 
cult algorithms as well as elegant, simple, and easy ones. Our challenge is to understand 
the former and to appreciate the latter in the context of scientific and commercial ap¬ 
plications. In doing so, we shall explore a variety of useful tools and develop a style of 
algorithmic thinking that will serve us well in computational challenges to come. 
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our study of algorithms is based upon implementing them as programs written in 
the Java programming language. We do so for several reasons: 

■ Our programs are concise, elegant, and complete descriptions of algorithms. 

■ You can run the programs to study properties of the algorithms. 

■ You can put the algorithms immediately to good use in applications. 

These are important and significant advantages over the alternatives of working with 
English-language descriptions of algorithms. 

A potential downside to this approach is that we have to work with a specific pro¬ 
gramming language, possibly making it difficult to separate the idea of the algorithm 
from the details of its implementation. Our implementations are designed to mitigate 
this difficulty, by using programming constructs that are both found in many modern 
languages and needed to adequately describe the algorithms. 

We use only a small subset of Java. While we stop short of formally defining the 
subset that we use, you will see that we make use of relatively few Java constructs, and 
that we emphasize those that are found in many modern programming languages. The 
code that we present is complete, and our expectation is that you will download it and 
execute it, on our test data or test data of your own choosing. 

We refer to the programming constructs, software libraries, and operating system 
features that we use to implement and describe algorithms as our programming model. 
In this section and section 1 . 2 , we fully describe this programming model. The treat¬ 
ment is self-contained and primarily intended for documentation and for your refer¬ 
ence in understanding any code in the book. The model we describe is the same model 
introduced in our book An Introduction to Programming in Java: An Interdisciplinary 
Approach, which provides a slower-paced introduction to the material. 

For reference, the figure on the facing page depicts a complete Java program that 
illustrates many of the basic features of our programming model. We use this code for 
examples when discussing language features, but defer considering it in detail to page 
46 (it implements a classic algorithm known as binary search and tests it for an applica¬ 
tion known as whitelist filtering) . We assume that you have experience programming 
in some modern language, so that you are likely to recognize many of these features in 
this code. Page references are included in the annotations to help you find answers to 
any questions that you might have. Since our code is somewhat stylized and we strive 
to make consistent use of various Java idioms and constructs, it is worthwhile even for 
experienced Java programmers to read the information in this section. 
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import a Java library (see page 27) 



system passes argument value 
"whitelist.txt" fomainQ 


command line 
(see page 36) s 


StdOut — 

(see page 37) 


filename (args[0]) 

I 

% java BinarySearch largeW.txt < largeT.txt 
► 499569 , t 

_ „ file redirectd from St din 

984875 (seepage 40) 


Anatomy of a Java program and its invocation from the command line 
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Basic Structure of a Java program A Java program (class) is either a library of 
static methods (functions) or a data type definition. To create libraries of static methods 
and data-type definitions, we use the following five components, the basis of program¬ 
ming in Java and many other modern languages: 

■ Primitive data types precisely define the meaning of terms like integer, real num¬ 
ber, and boolean value within a computer program. Their definition includes the 
set of possible values and operations on those values, which can be combined 
into expressions like mathematical expressions that define values. 

■ Statements allow us to define a computation by creating and assigning values to 
variables, controlling execution flow, or causing side effects. We use six types of 
statements: declarations, assignments, conditionals, loops, calls, and returns. 

■ Arrays allow us to work with multiple values of the same type. 

■ Static methods allow us to encapsulate and reuse code and to develop programs 
as a set of independent modules. 

■ Strings are sequences of characters. Some operations on them are built in to Java. 

■ Input/output sets up communication between programs and the outside world. 

■ Data abstraction extends encapsulation and reuse to allow us to define non¬ 
primitive data types, thus supporting object-oriented programming. 

In this section, we will consider the first five of these in turn. Data abstraction is the 
topic of the next section. 

Running a Java program involves interacting with an operating system or a program 
development environment. For clarity and economy, we describe such actions in terms 
of a virtual terminal, where we interact with programs by typing commands to the 
system. See the booksite for details on using a virtual terminal on your system, or for 
information on using one of the many more advanced program development environ¬ 
ments that are available on modern systems. 

For example, BinarySearch is two static methods, rank() and main(). The first 
static method, rank(), is four statements: two declarations, a loop (which is itself an as¬ 
signment and two conditionals), and a return. The second, mai n (), is three statements: 
a declaration, a call, and a loop (which is itself an assignment and a conditional). 

To invoke a Java program, we first compile it using the j avac command, then run it us¬ 
ing the java command. For example, to run Bi narySearch, we first type the command 
javac Bi narySearch. java (which creates a file Bi narySearch. cl ass that contains 
a lower-level version of the program in Java bytecode in the file Bi narySearch. cl ass). 
Then we type java Bi narySearch (followed by a whitelist file name) to transfer con¬ 
trol to the bytecode version of the program. To develop a basis for understanding the 
effect of these actions, we next consider in detail primitive data types and expressions, 
the various kinds of Java statements, arrays, static methods, strings, and input/output. 
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Primitive data types and expressions A data type is a set of values and a set of 

operations on those values. We begin by considering the following four primitive data 
types that are the basis of the Java language: 

■ Integers, with arithmetic operations (i nt) 

■ Real numbers, again with arithmetic operations (doubl e) 

■ Booleans, the set of values { true, false } with logical operations (bool ean) 

■ Characters, the alphanumeric characters and symbols that you type (char) 

Next we consider mechanisms for specifying values and operations for these types. 

A Java program manipulates variables that are named with identifiers. Each variable 
is associated with a data type and stores one of the permissible data-type values. In Java 
code, we use expressions like familiar mathematical expressions to apply the operations 
associated with each type. For primitive types, we use identifiers to refer to variables, 
operator symbols such as + - * /to specify operations, literals such as 1 or 3.14 to 
specify values, and expressions such as (x + 2.236)/2 to specify operations on values. 
The purpose of an expression is to define one of the data-type values. 


term 


examples 

definition 

primitive 
data type 



a set of values and a set of 

i nt 

double boolean char 

operations on those values 
(built in to the Java language) 

a sequence of letters, digits, 

identifier 

a abc 

Ab$ a„b abl23 lo hi 

_, and $, the first of which is 
not a digit 

variable 


[any identifier ] 

names a data-type value 

operator 

i nt 

+ - * / 

1 0 -42 

names a data-type operation 

literal 

double 

2.0 1.0e-15 3.14 

source-code representation 

boolean 

char 

true false 

'a' '+' ’9' ’\n' 

of a value 


int 

lo + (hi - 1 o)/2 

a literal, a variable, or a 

expression 


1.0e-15 * t 

sequence of operations on 

boolean 

lo <= hi 

literals and/or variables that 
produces a value 


Basic building blocks for Java programs 
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To define a data type, we need only specify the values and the set of operations on 
those values. This information is summarized in the table below for Java’s i nt, doubl e, 
bool ean, and char data types. These data types are similar to the basic data types found 
in many programming languages. For int and double, the operations are familiar 
arithmetic operations; for bool ean, they are familiar logical operations. It is important 
to note that +, -, *, and / are overloaded —the same symbol specifies operations in mul¬ 
tiple different types, depending on context. The key property of these primitive opera¬ 
tions is that an operation involving values of a given type has a value of that type. This rule 
highlights the idea that we are often working with approximate values, since it is often 
the case that the exact value that would seem to be defined by the expression is not a 
value of the type. For example, 5/3 has the value 1 and 5.0/3.0 has a value very close 
to 1.66666666666667 but neither of these is exactly equal to 5/3. This table is far from 
complete; we discuss some additional operators and various exceptional situations that 
we occasionally need to consider in the Q&A at the end of this section. 



set of values 


typical expressions 

type 

operators 

expression 

value 


i nt 

integers between 

2 31 and+2 31 — 1 
(32-bit two’s 
complement) 

+ (add) 

- (subtract) 

* (multiply) 

/ (divide) 

% (remainder) 

5 + 3 

5 - 3 

5 * 3 

5 / 3 

5 % 3 

8 

2 

15 

1 

2 


double-precision 

+ (add) 

3.141 - .03 

3.111 

double 

real numbers 

- (subtract) 

2.0 - 2.Oe-7 

1.9999998 


(64-bit IEEE 754 

* (multiply) 

100 * .015 

1.5 


standard) 

/ (divide) 

6.02e23 / 2.0 

3.01e23 



&& (and) 

true && false 

fal se 

boolean 

true or false 

1 1 (or) 

false | | true 

true 


! (not) 

!false 

true 



a (xor) 

true a true 

fal se 


char 


characters 

(16-bit) 


[arithmetic operations, rarely used ] 


Primitive data types in Java 
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Expressions. As illustrated in the table at the bottom of the previous page, typical ex¬ 
pressions are infix: a literal (or an expression), followed by an operator, followed by 
another literal (or another expression). When an expression contains more than one 
operator, the order in which they are applied is often significant, so the following pre¬ 
cedence conventions are part of the Java language specification: The operators * and / ( 
and %) have higher precedence than (are applied before) the + and - operators; among 
logical operators, ! is the highest precedence, followed by && and then | |. Generally, 
operators of the same precedence are applied left to right. As in standard arithmetic ex¬ 
pressions, you can use parentheses to override these rules. Since precedence rules vary 
slightly from language to language, we use parentheses and otherwise strive to avoid 
dependence on precedence rules in our code. 

Type conversion. Numbers are automatically promoted to a more inclusive type if no 
information is lost. For example, in the expression 1 + 2.5, the 1 is promoted to the 
double value 1.0 and the expression evaluates to the doubl e value 3.5 . A cast is a type 
name in parentheses within an expression, a directive to convert the following value 
into a value of that type. For example (i nt) 3.7 is 3 and (doubl e) 3 is 3.0. Note that 
casting to an i nt is truncation instead of rounding—rules for casting within compli¬ 
cated expressions can be intricate, and casts should be used sparingly and with care. A 
best practice is to use expressions that involve literals or variables of a single type. 

Comparisons. The following operators compare two values of the same type and 
produce a boolean value: equal (==), not equal (!=), less than (<), less than or equal 
(<=), greater than (>), and greater than or equal (>=). These operators are known as 
mixed-type operators because their value is bool ean, not the type of the values being 
compared. An expression with a boolean value is known as a boolean expression. Such 
expressions are essential components in conditional and loop statements, as we will see. 

Other primitive types. Java’s i nt has 2 32 different values by design, so it can be repre¬ 
sented in a 32-bit machine word (many machines have 64-bit words nowadays, but the 
32-bit int persists). Similarly, the double standard specifies a 64-bit representation. 
These data-type sizes are adequate for typical applications that use integers and real 
numbers. To provide flexibility, Java has five additional primitive data types: 

■ 64-bit integers, with arithmetic operations (1 ong) 

■ 16-bit integers, with arithmetic operations (short) 

■ 16-bit characters, with arithmetic operations (char) 

■ 8-bit integers, with arithmetic operations (byte) 

■ 32-bit single-precision real numbers, again with arithmetic operations (f 1 oat) 
We most often use i nt and doubl e arithmetic operations in this book, so we do not 
consider the others (which are very similar) in further detail here. 
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Statements A Java program is composed of statements, which define the computa¬ 
tion by creating and manipulating variables, assigning data-type values to them, and 
controlling the flow of execution of such operations. Statements are often organized in 
blocks, sequences of statements within curly braces. 

■ Declarations create variables of a specified type and name them with identifiers. 

■ Assignments associate a data-type value (defined by an expression) with a vari¬ 
able. Java also has several implicit assignment idioms for changing the value of a 
data-type value relative to its current value, such as incrementing the value of an 
integer variable. 

■ Conditionals provide for a simple change in the flow of execution—execute the 
statements in one of two blocks, depending on a specified condition. 

■ Loops provide for a more profound change in the flow of execution—execute the 
statements in a block as long as a given condition is true. 

■ Calls and returns relate to static methods (see page 22), which provide another way 
to change the flow of execution and to organize code. 

A program is a sequence of statements, with declarations, assignments, conditionals, 
loops, calls, and returns. Programs typically have a nested structure: a statement among 
the statements in a block within a conditional or a loop may itself be a conditional or a 
loop. For example, the whi 1 e loop in rankO contains an if statement. Next, we con¬ 
sider each of these types of statements in turn. 

Declarations. A declaration statement associates a variable name with a type at com¬ 
pile time. Java requires us to use declarations to specify the names and types of vari¬ 
ables. By doing so, we are being explicit about any computation that we are specify¬ 
ing. Java is said to be a strongly typed language, because the Java compiler checks for 
consistency (for example, it does not permit us to multiply a bool ean and a doubl e). 
Declarations can appear anywhere before a variable is first used—most often, we put 
them at the point of first use. The scope of a variable is the part of the program where it 
is defined. Generally the scope of a variable is composed of the statements that follow 
the declaration in the same block as the declaration. 

Assignments. An assignment statement associates a data-type value (defined by an ex¬ 
pression) with a variable. When we write c = a + b in Java, we are not expressing 
mathematical equality, but are instead expressing an action: set the value of the vari¬ 
able c to be the value of a plus the value of b. It is true that c is mathematically equal 
to a + b immediately after the assignment statement has been executed, but the point 
of the statement is to change the value of c (if necessary). The left-hand side of an as¬ 
signment statement must be a single variable; the right-hand side can be an arbitrary 
expression that produces a value of the type. 
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Conditionals. Most computations require different actions for different inputs. One 
way to express these differences in Java is the i f statement: 

if (<boolean expression:*) { <b1ock statements> } 

This description introduces a formal notation known as a template that we use occa¬ 
sionally to specify the format of Java constructs. We put within angle brackets (< >) 
a construct that we have already defined, to indicate that we can use any instance of 
that construct where specified. In this case, <boolean expression> represents an 
expression that has a boolean value, such as one involving a comparison operation, 
and <block statements> represents a sequence of Java statements. It is possible to 
make formal definitions of <boolean expression> and cblock statements:*, but 
we refrain from going into that level of detail. The meaning of an i f statement is self- 
explanatory: the statement(s) in the block are to be executed if and only if the boolean 
expression is true. The i f-el se statement: 

if (<boolean expression:*) { <bfock statements> } 
else { <block statements> } 

allows for choosing between two alternative blocks of statements. 

Loops. Many computations are inherently repetitive. The basic Java construct for han¬ 
dling such computations has the following format: 

while (<boolean expression>) { <block statements> } 

The whi 1 e statement has the same form as the i f statement (the only difference being 
the use of the keyword while instead of i f), but the meaning is quite different. It is an 
instruction to the computer to behave as follows: if the boolean expression is false, 
do nothing; if the boolean expression is true, execute the sequence of statements in 
the block (just as with i f) but then check the boolean expression again, execute the se¬ 
quence of statements in the block again if the boolean expression is true, and continue 
as long as the boolean expression is true. We refer to the statements in the block in a 
loop as the body of the loop. 

Break and continue. Some situations call for slightly more complicated control flow 
than provide by the basic i f and whi 1 e statements. Accordingly, Java supports two ad¬ 
ditional statements for use within while loops: 

■ The break statement, which immediately exits the loop 

■ The conti nue statement, which immediately begins the next iteration of the 
loop 

We rarely use these statements in the code in this book (and many programmers never 
use them), but they do considerably simplify code in certain instances. 
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Shortcut notations There are several ways to express a given computation; we 
seek clear, elegant, and efficient code. Such code often takes advantage of the following 
widely used shortcuts (that are found in many languages, not just Java). 

Initializing declarations. We can combine a declaration with an assignment to ini¬ 
tialize a variable at the same time that it is declared (created). For example, the code 
i nt i = 1; creates an i nt variable named i and assigns it the initial value 1. A best 
practice is to use this mechanism close to first use of the variable (to limit scope). 

Implicit assignments. The following shortcuts are available when our purpose is to 
modify a variable’s value relative to its current value: 

■ Increment/decrement operators: i ++ is the same as i = i +1 and has the value 
i in an expression. Similarly, i — is the same as i = i - 1. The code ++i and 
--i are the same except that the expression value is taken after the increment/ 
decrement, not before. 

■ Other compound operations: Prepending a binary operator to the = in an as¬ 
signment is equivalent to using the variable on the left as the first operand. 

For example, the code i /=2; is equivalent to the code i = i /2 ; Note that 
i += 1 ; has the same effect as i = i+1; (andi++). 

Single-statement blocks. If a block of statements in a conditional or a loop has only a 
single statement, the curly braces may be omitted. 

For notation. Many loops follow this scheme: initialize an index variable to some val¬ 
ue and then use a whi 1 e loop to test a loop continuation condition involving the index 
variable, where the last statement in the whi 1 e loop increments the index variable. You 
can express such loops compactly with Java’s for notation: 

for (<initialize>; <boolean expressions <increment>) 

{ 

<block statements> 

} 

This code is, with only a few exceptions, equivalent to 
<i ni ti al i ze> ; 

while (<boolean expression>) 

{ 

<block statements> 

<increment>; 

} 

We use for loops to support this initialize-and-increment programming idiom. 
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statement 

examples definition 

declaration 

1 nt 1 ’ create a variable of a specified type, 

double c; named with a given identifier 

assignment 

a = b + 3; 

,. . . , , assign a data-type value to a variable 

discriminant = b*b - 4.0*c; 

initializing 

declaration 

int i = 1; declaration that also assigns an 

double c = 3.14159262 5; initial value 

implicit 

assignment 

i++; 

i = i +1; 

i += 1 ; 

conditional (i f) 

execute a statement, 

i -j- rx < 0) x = — x ■ 7 

depending on boolean expression 

conditional 
(i f-el se) 

if (x > y) max = x; execute one or the other statement, 

else max = y; depending on boolean expression 

loop (whi 1 e) 

int v = 0; 
while (v <= N) 

v = 2 " v ! execute statement 

double t = c; until boolean expression is fal se 

while (Math.abs(t - c/t) > le-15*t) 
t = (c/t + t) / 2.0; 

loop (for) 

for (int i =1; i <= N; i++) 
sum += 1.0/i; 

compact version of whi 1 e statement 

for (int i =0; i <= N; i++) 

StdOut.println(2*Math.PI*i/N); 

call 

int key = Stdln. readlnt() ; invoke other methods (see page 22) 

return 

return false; return from a method (see page 24) 

Java statements 
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Arrays An array stores a sequence of values that are all of the same type. We want 
not only to store values but also to access each individual value. The method that we 
use to refer to individual values in an array is numbering and then indexing them. If 
we have N values, we think of them as being numbered from 0 to N~ 1. Then, we can 
unambiguously specify one of them in Java code by using the notation a [i ] to refer to 
the ith value for any value of i from 0 to N-l. This Java construct is known as a one¬ 
dimensional array. 

Creating and initializing an array. Making an array in a Java program involves three 
distinct steps: 

■ Declare the array name and type. 

■ Create the array. 

■ Initialize the array values. 

To declare the array, you need to specify a name and the type of data it will contain. 
To create it, you need to specify its length (the number of values). For example, the 
“long form” code shown at right makes 
an array of N numbers of type doubl e, all 
initialized to 0.0. The first statement is 
the array declaration. It is just like a dec¬ 
laration of a variable of the correspond¬ 
ing primitive type except for the square 
brackets following the type name, which 
specify that we are declaring an array. 

The keyword new in the second state¬ 
ment is a Java directive to create the ar¬ 
ray. The reason that we need to explicitly 
create arrays at run time is that the Java 
compiler cannot know how much space 
to reserve for the array at compile time (as it can for primitive-type values). The for 
statement initializes the N array values. This code sets all of the array entries to the value 
0.0. When you begin to write code that uses an array, you must be sure that your code 
declares, creates, and initializes it. Omitting one of these steps is a common program¬ 
ming mistake. 

Short form. For economy in code, we often take advantage of Java’s default array ini¬ 
tialization convention and combine all three steps into a single statement, as in the 
“short form” code in our example. The code to the left of the equal sign constitutes the 
declaration; the code to the right constitutes the creation. The f o r loop is unnecessary 
in this case because the default initial value of variables of type doubl e in a Java array is 


|on g form declaration 

doubl e[] a; creation 

a = new doubleJN];^ 
for (int i = 0; i < N; i++) 
a[i] = 0.0; 

^ initialization 

short form 

doubled a = new doubleJN]; 

initializing declaration 

int[] a = { 1, 1, 2, 3, 5, 8 }; 


Declaring, creating and initializing an array 
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0.0, but it would be required if a nonzero value were desired. The default initial value 
is zero for numeric types and fal se for type bool ean. The third option shown for our 
example is to specify the initialization values at compile time, by listing literal values 
between curly braces, separated by commas. 

Using an array. Typical array-processing code is shown on page 21. After declaring 
and creating an array, you can refer to any individual value anywhere you would use 
a variable name in a program by enclosing an integer index in square brackets after 
the array name. Once we create an array, its size is fixed. A program can refer to the 
length of an array a[] with the code a. length. The last element of an array a[] is 
always a [a. 1 ength-1] . Java does automatic bounds checking —if you have created an 
array of size N and use an index whose value is less than 0 or greater than N-l, your pro¬ 
gram will terminate with an ArrayOutOfBoundsExcepti on runtime exception. 

Aliasing. Note carefully that an array name refers to the whole array —if we assign one 
array name to another, then both refer to the same array, as illustrated in the following 
code fragment. 

int[] a = new int [N] ; 

a[i] = 1234; 

int[] b = a; 

b[i] = 5678; // a[i] is now 5678. 

This situation is known as aliasing and can lead to subtle bugs. If your intent is to make 
a copy of an array, then you need to declare, create, and initialize a new array and then 
copy all of the entries in the original array to the new array, as in the third example on 
page 21. 

Two-dimensional arrays. A two-dimensional array in Java is an array of one-dimen¬ 
sional arrays. A two-dimensional array may be ragged (its arrays may all be of differing 
lengths), but we most often work with (for appropriate parameters M and N) M-by-N 
two-dimensional arrays that are arrays of M rows, each an array of length N (so it also 
makes sense to refer to the array as having N columns). Extending Java array constructs 
to handle two-dimensional arrays is straightforward. To refer to the entry in row i and 
column j of a two-dimensional array a [] [], we use the notation a [i ] [ j ]; to declare a 
two-dimensional array, we add another pair of square brackets; and to create the array, 
we specify the number of rows followed by the number of columns after the type name 
(both within square brackets), as follows: 
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double[][] a = new double[M] [N]; 

We refer to such an array as an M-by-iV array. By convention, the first dimension is the 
number of rows and the second is the number of columns. As with one-dimensional 
arrays, Java initializes all entries in arrays of numeric types to zero and in arrays of 
boolean values to false. Default initialization of two-dimensional arrays is useful 
because it masks more code than for one-dimensional arrays. The following code is 
equivalent to the single-line create-and-initialize idiom that we just considered: 

doublet] [] a; 
a = new double[M]tN] ; 
for (int i =0; i < M; i++) 
for (int j =0; j < N; j++) 
ati] tj] = 0.0; 

This code is superfluous when initializing to zero, but the nested f o r loops are needed 
to initialize to other value(s). 
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task 


implementation (code fragment) 


find the maximum of 
the array values 


double max = a[0]; 
for (int i =1; i < a. length; i++) 
if (a[i] > max) max = a[i]; 


compute the average of 
the array values 


int N = a.length; 
double sum = 0.0; 
for (int i = 0; i < N; i++) 
sum += a[i ] ; 

double average = sum / N; 


copy to another array 


int N = a.length; 
doublet] b = new double[N] ; 
for (int i = 0; i < N; i++) 
b[i ] = a[i] ; 


reverse the elements 
within an array 


int N = a.length; 
for (int i =0; i < N/2; i++) 
{ 

double temp = a[i] ; 
a[i] = a[N-l-i] ; 
a[N-i-l] = temp; 

} 


matrix-matrix multiplication 
(square matrices ) 

a[][]*b[][] = c[][] 


int N = a. length; 
doublet] [] c = new double[N] [N] ; 
for (int i =0; i < N; i++) 
for (int j = 0; j < N; j++) 

{ // Compute dot product of row i and column j. 
for (int k = 0; k < N; k++) 
c[i] [j] += a[i] [k]*btk] [j]; 

} 


Typical array-processing code 
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Static methods Every Java program in this book is either a data-type definition 
(which we describe in detail in section 1 . 2 ) or a library of static methods (which we de¬ 
scribe here). Static methods are called functions in many programming languages, since 
they can behave like mathematical functions, as described next. Each static method is 
a sequence of statements that are executed, one after the other, when the static method 
is called, in the manner described below. The modifier static distinguishes these meth¬ 
ods from instance methods, which we discuss in section 1 . 2 . We use the word method 
without a modifier when describing characteristics shared by both kinds of methods. 


Defining a static method. A method encapsulates a computation that is defined as a 
sequence of statements. A method takes arguments (values of given data types) and 
computes a return value of some data type that depends upon the arguments (such 
as a value defined by a mathematical function) or causes a side effect that depends on 
the arguments (such as printing a value). The static method rank() in Bi narySearch 

is an example of the first; mai n () is an ex- 


argument 

variable 


signature return method argument 

\ type name type 

public static |doub1e||sqrt| ( [double cl ) | 


{ 


local 

variables ~ 

method _ 
body 


if (c < 0) 
double err 
Idoubl e 


whi 1 e 
t : 


_(c/t 

return t; I 

— \ 


return Double.NaN; 

= le-15; 

c; _ 

|(Math.abs(t - c/t)| > err 


t) 


+ t) / 2.0; 


return statement 


call on another method 


Anatomy of a static method 


ample of the second. Each static method 
is composed of a signature (the keywords 
publ i c stati c followed by a return type, 
the method name, and a sequence of ar¬ 
guments, each with a declared type) and 
a body (a statement block: a sequence of 
statements, enclosed in curly braces). Ex¬ 
amples of static methods are shown in the 
table on the facing page. 

Invoking a static method. A call on a static 
method is its name followed by expressions 
that specify argument values in parenthe¬ 
ses, separated by commas. When the method call is part of an expression, the method 
computes a value and that value is used in place of the call in the expression. For ex¬ 
ample the call on rank() in BinarySearchO returns an int value. A method call 
followed by a semicolon is a statement that generally causes side effects. For example, 
the call Arrays. sort() in main() in Bi narySearch is a call on the system method 
Arrays.sortO that has the side effect of putting the entries in the array in sorted 
order. When a method is called, its argument variables are initialized with the values 
of the corresponding expressions in the call. A return statement terminates a static 
method, returning control to the caller. If the static method is to compute a value, that 
value must be specified in a return statement (if such a static method can reach the 
end of its sequence of statements without a return, the compiler will report the error). 
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task 

implementation 

absolute value of an 
i nt value 

public static int abs(int x) 

{ 

if (x < 0) return -x; 
else return x; 

} 

absolute value of a 
doubl e value 

public static double abs(double x) 

{ 

if (x < 0.0) return -x; 
else return x; 

} 

primality test 

public static boolean isPrime(int N) 

{ 

if (N < 2) return false; 
for (int i = 2; i*i <= N; i++) 
if (N % i == 0) return false; 
return true; 

} 

square root 
(Newton’s method) 

public static double sqrt(double c) 

{ 

if (c > 0) return Double.NaN; 
double err = le-15; 
double t = c; 

while (Math.abs(t - c/t) > err * t) 
t = (c/t + t) / 2.0; 
return t; 

} 

hypotenuse of 
a right triangle 

public static double hypotenuse(double a, double b) 

{ return Math.sqrt(a*a + b*b); } 

Harmonic number 
(see page 185) 

public static double H(int N) 

{ 

double sum = 0.0; 
for (int i = 1; i <= N; i++) 
sum += 1.0 / i; 
return sum; 

} 

Typical implementations of static methods 
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Properties of methods. A complete detailed description of the properties of methods 
is beyond our scope, but the following points are worth noting: 

■ Arguments are passed by value. You can use argument variables anywhere in the 
code in the body of the method in the same way you use local variables. The 
only difference between an argument variable and a local variable is that the 
argument variable is initialized with the argument value provided by the call¬ 
ing code. The method works with the value of its arguments, not the arguments 
themselves. One consequence of this approach is that changing the value of an 
argument variable within a static method has no effect on the calling code. Gen¬ 
erally, we do not change argument variables in the code in this book. The pass- 
by-value convention implies that array arguments are aliased (see page 19)—the 
method uses the argument variable to refer to the caller’s array and can change 
the contents of the array (though it cannot change the array itself). For example, 
Arrays. sort() certainly changes the contents of the array passed as argument: 
it puts the entries in order. 

■ Method names can be overloaded. For example, the Java Math library uses 
this approach to provide implementations of Math. abs(), Math .mi n(), and 
Math . max() for all primitive numeric types. Another common use of overload¬ 
ing is to define two different versions of a function, one that takes an argument 
and another that uses a default value of that argument. 

■ A method has a single return value but may have multiple return statements. A 
Java method can provide only one return value, of the type declared in the 
method signature. Control goes back to the calling program as soon as the first 
return statement in a static method is reached. You can put return statements 
wherever you need them. Even though there may be multiple retu rn statements, 
any static method returns a single value each time it is invoked: the value follow¬ 
ing the first return statement encountered. 

■ A method can have side effects. A method may use the keyword voi d as its return 
type, to indicate that it has no return value. An explicit return is not necessary 
in a voi d static method: control returns to the caller after the last statement. 

A voi d static method is said to produce side effects (consume input, produce 
output, change entries in an array, or otherwise change the state of the system). 
For example, the mai n() static method in our programs has a voi d return type 
because its purpose is to produce output. Technically, voi d methods do not 
implement mathematical functions (and neither does Math. random (), which 
takes no arguments but does produce a return value). 

The instance methods that are the subject of section 2.1 share these properties, though 
profound differences surround the issue of side effects. 


www.it-ebooks.info 



1.1 ■ Basic Programming Model 


25 


Recursion. A method can call itself (if you are not comfortable with this idea, known 
as recursion, you are encouraged to work exercises 1 . 1.16 through 1 . 1 . 22 ). For ex¬ 
ample, the code at the bottom of this page gives an alternate implementation of the 
rank() method in Bi narySearch. We often use recursive implementations of methods 
because they can lead to compact, elegant code that is easier to understand than a cor¬ 
responding implementation that does not use recursion. For example, the comment 
in the implementation below provides a succinct description of what the code is sup¬ 
posed to do. We can use this comment to convince ourselves that it operates correctly, 
by mathematical induction. We will expand on this topic and provide such a proof for 
binary search in section 3 . 1 . There are three important rules of thumb in developing 
recursive programs: 

■ The recursion has a base case —we always include a conditional statement as the 
first statement in the program that has a return. 

■ Recursive calls must address subproblems that are smaller in some sense, so 
that recursive calls converge to the base case. In the code below, the difference 
between the values of the fourth and the third arguments always decreases. 

■ Recursive calls should not address subproblems that overlap. In the code below, 
the portions of the array referenced by the two subproblems are disjoint. 

Violating any of these guidelines is likely to lead to incorrect results or a spectacularly 
inefficient program (see exercises 1 . 1.19 and 1 . 1 . 27 ). Adhering to them is likely to 
lead to a clear and correct program whose performance is easy to understand. Another 
reason to use recursive methods is that they lead to mathematical models that we can 
use to understand performance. We address this issue for binary search in section 3.2 
and in several other instances throughout the book. 


public static int rank(int key, int[] a) 

{ return rank(key, a, 0, a.length -1); } 

public static int rank(int key, int[] a, int lo, int hi) 

{ // Index of key in a[], if present, is not smaller than lo 


// 


and not larger than hi. 


if (lo > hi) 
int mid = lo 
if (key 

else if (key 
el se 


return -1; 

+ (hi - lo) / 2; 

< a[mid]) return rank(key, a, lo, mid - 1); 
> a[mid]) return rank(key, a, mid + 1, hi); 


return mid; 


} 


Recursive implementation of binary search 
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Basic programming model. A library of static methods is a set of static methods that 
are defined in a Java class, by creating a file with the keywords pub! i c cl ass followed 
by the class name, followed by the static methods, enclosed in braces, kept in a file with 
the same name as the class and a . j ava extension. A basic model for Java programming 
is to develop a program that addresses a specific computational task by creating a li¬ 
brary of static methods, one of which is named mai n (). Typing j ava followed by a class 
name followed by a sequence of strings leads to a call on mai n () in that class, with an 
array containing those strings as argument. After the last statement in mai n () executes, 
the program terminates. In this book, when we talk of a Java program for accomplishing 
a task, we are talking about code developed along these lines (possibly also including a 
data-type definition, as described in section 1 . 2 ). For example, Bi narySearch is a Java 
program composed of two static methods, rank() and mai n(), that accomplishes the 
task of printing numbers on an input stream that are not found in a whitelist file given 
as command-line argument. 

Modular programming. Of critical importance in this model is that libraries of stat¬ 
ic methods enable modular programming where we build libraries of static methods 
(modules ) and a static method in one library can call static methods defined in other 
libraries. This approach has many important advantages. It allows us to 

■ Work with modules of reasonable size, even in program involving a large 
amount of code 

■ Share and reuse code without having to reimplement it 

■ Easily substitute improved implementations 

■ Develop appropriate abstract models for addressing programming problems 

■ Localize debugging (see the paragraph below on unit testing) 

For example, Bi narySearch makes use of three other independently developed librar¬ 
ies, our Stdln and In library and Java’s Arrays library. Each of these libraries, in turn, 
makes use of several other libraries. 

Unit testing. A best practice in Java programming is to include a mai n() in every li¬ 
brary of static methods that tests the methods in the library (some other programming 
languages disallow multiple mai n() methods and thus do not support this approach). 
Proper unit testing can be a significant programming challenge in itself. At a minimum, 
every module should contain a mai n() method that exercises the code in the module 
and provides some assurance that it works. As a module matures, we often refine the 
mai n () method to be a development client that helps us do more detailed tests as we 
develop the code, or a test client that tests all the code extensively. As a client becomes 
more complicated, we might put it in an independent module. In this book, we use 
mai n Q to help illustrate the purpose of each module and leave test clients for exercises. 
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External libraries. We use static methods from four different kinds of libraries, each 
requiring (slightly) differing procedures for code reuse. Most of these are libraries of 
static methods, but a few are data-type definitions that also include some static methods. 

■ The standard system libraries j ava. 1 ang . *. These include Math, which contains 
methods for commonly used mathematical functions; Integer and Doubl e, 
which we use for converting between strings of characters and 
i nt and doubl e values; Stri ng and Stri ngBui 1 der, which 
we discuss in detail later in this section and in chapter 5 ; and 
dozens of other libraries that we do not use. 

■ Imported system libraries such as java, uti 1 .Arrays. There 
are thousands of such libraries in a standard Java release, but 
we make scant use of them in this book. An i mport statement 
at the beginning of the program is needed to use such libraries 
(and signal that we are doing so). 

■ Other libraries in this book. For example, another program can 
use rank() in Bi narySearch. To use such a program, down¬ 
load the source from the booksite into your working directory. 

■ The standard libraries Std* that we have developed for use 
in this book (and our introductory book An Introduction to 
Programming in Java: An Interdisciplinary Approach). These 
libraries are summarized in the following several pages. Source 
code and instructions for downloading them are available on 
the booksite. 

To invoke a method from another library (one in the same directory 
or a specified directory, a standard system library, or a system library 
that is named in an i mport statement before the class definition), we 
prepend the library name to the method name for each call. For ex¬ 
ample, the mai n() method in Bi narySearch calls the sort() method 
in the system library java, uti 1 .Arrays, the readlntsO method in 
our library In, and the pri ntl n() method in our library StdOut. 


standard system libraries 
Math 
Integer 1- 
Double 1 
Stri ng + 

StringBui1der 
System 

imported system libraries 
java.util.Arrays 
our standard libraries 
Stdln 
StdOut 
StdDraw 
StdRandom 
StdStats 
In 1 
Out 1 

1 data type definitions that 
include some static methods 

Libraries with static 
methods used in this book 


LIBRARIES OF METHODS IMPLEMENTED BY OURSELVES AND BY OTHERS in a modular 
programming environment can vastly expand the scope of our programming model. 
Beyond all of the libraries available in a standard Java release, thousands more are avail¬ 
able on the web for applications of all sorts. To limit the scope of our programming 
model to a manageable size so that we can concentrate on algorithms, we use just the 
libraries listed in the table at right on this page, with a subset of their methods listed in 
APIs, as described next. 
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APIs A critical component of modular programming is documentation that explains 
the operation of library methods that are intended for use by others. We will consis¬ 
tently describe the library methods that we use in this book in application programming 
interfaces (APIs) that list the library name and the signatures and short descriptions of 
each of the methods that we use. We use the term client to refer to a program that calls 
a method in another library and the term implementation to describe the Java code that 
implements the methods in an API. 

Example. The following example, the API for commonly used static methods from the 
standard Math library in j ava. 1 ang, illustrates our conventions for APIs: 


public class Math 

static double abs (double a) absolute value of a 

static double max (double a, double b) maximum of a and b 

static double min (double a, double b) minimum of a and b 

Note 1: abs(), max(), and mi n() are defined also for i nt, long, and float, 
static double sin (double theta) sine function 

static double cos (double theta) cosine function 

static double tan (double theta) tangent function 

Note 2: Angles are expressed in radians. Use toDeg rees () and toRadi ans () to convert. 

Note 3: Use asi n (), acos (), and atan () for inverse functions. 

static double exp(double a) exponential (e a ) 

static double log (double a) natural log (log e a, or In a) 

static double pow(double a, double b) raise a to the bth power (a b ) 


static double random() random number in [0, 1) 

static double sqrt (double a) square root of a 


stati c doubl e E value ofe (constant) 

static double PI value of tt (constant) 

See booksitefor other available functions. 

API for Java's mathematics library (excerpts) 
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These methods implement mathematical functions—they use their arguments to com¬ 
pute a value of a specified type (except random (), which does not implement a math¬ 
ematical function because it does not take an argument). Since they all operate on 
double values and compute a double result, you can consider them as extending the 
double data type—extensibility of this nature is one of the characteristic features of 
modern programming languages. Each method is described by a line in the API that 
specifies the information you need to know in order to use the method. The Math li¬ 
brary also defines the precise constant values PI (for tt) and E (for e), so that you can 
use those names to refer to those constants in your programs. For example, the value 
of Math . si n (Math . PI/2) is 1.0 and the value of Math. log (Math . E) is 1. 0 (because 
Math. si n() takes its argument in radians and Math.logO implements the natural 
logarithm function). 

Java libraries. Extensive online descriptions of thousands of libraries are part of every 
Java release, but we excerpt just a few methods that we use in the book, in order to clear¬ 
ly delineate our programming model. For example, BinarySearch uses the sort() 
method from Java’s Arrays library, which we document as follows: 


public class Arrays 

stati c voi d sort(i nt [] a) put the array in increasing order 

Note: This method is defined also for other primitive types and Object. 

Excerpt from Java's Arrays library (java.util .Arrays) 


The Arrays library is not in java. 1 ang, so an import statement is needed to use it, as 
in Bi narySearch. Actually, chapter 2 of this book is devoted to implementations of 
sortO for arrays, including the mergesort and quicksort algorithms that are imple¬ 
mented in Arrays. sort(). Many of the fundamental algorithms that we consider in 
this book are implemented in Java and in many other programming environments. For 
example, Arrays also includes an implementation of binary search. To avoid confusion, 
we generally use our own implementations, although there is nothing wrong with using 
a finely tuned library implementation of an algorithm that you understand. 
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Our standard libraries. We have developed a number of libraries that provide useful 
functionality for introductory Java programming, for scientific applications, and for 
the development, study, and application of algorithms. Most of these libraries are for 
input and output; we also make use of the following two libraries to test and analyze 
our implementations. The first extends Math. randomO to allow us to draw random 
values from various distributions; the second supports statistical calculations: 


public class StdRandom 


static void initialize (long seed) 

static double randomO 

static int uniform(int N) 

static int uniform(int lo, int hi) 

static double uniform(double lo, double hi) 

static boolean bernoul1i(double p) 

static double gaussianO 

static double gaussian(double m, double s) 
static int discrete(double[] a) 


initialize 

real between 0 and 1 
integer between 0 and N-l 
integer between 1 o and hi -1 
real between 1 o and hi 
true with probability p 
normal, mean 0, std dev 1 
normal, mean m, std dev s 
i with probability a [i ] 
randomly shuffle the array a [] 


static void shuffle(double[] a) 

Note: overloaded implementations o/shuffleO are included for other primitive types and for Object. 

API for our library of static methods for random numbers 


public class StdStats 


static double max(double[] a) 
static double min (doublet] a) 
static double mean (doublet] a) 
static double var(double[] a) 
static double stddev(doublet] a) 
static double median(double[] a) 


largest value 
smallest value 
average 

sample variance 

sample standard deviation 

median 


API for our library of static methods for data analysis 
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The i ni ti al i ze() method in StdRandom allows us to seed the random number gen¬ 
erator so that we can reproduce experiments involving random numbers. For reference, 
implementations of many of these methods are given on page 32. Some of these methods 
are extremely easy to implement; why do we bother including them in a library? An¬ 
swers to this question are standard for well-designed libraries: 

■ They implement a level of abstraction that allow us to focus on implement¬ 
ing and testing the algorithms in the book, not generating random objects or 
calculating statistics. Client code that uses such methods is clearer and easier to 
understand than homegrown code that does the same calculation. 

■ Library implementations test for exceptional conditions, cover rarely encoun¬ 
tered situations, and submit to extensive testing, so that we can count on them to 
operate as expected. Such implementations might involve a significant amount 
of code. For example, we often want implementations for various types of data. 
For example, Java’s Arrays library includes multiple overloaded implementa¬ 
tions of sort(), one for each type of data that you might need to sort. 

These are bedrock considerations for modular programming in Java, but perhaps a bit 
overstated in this case. While the methods in both of these libraries are essentially self- 
documenting and many of them are not difficult to implement, some of them represent 
interesting algorithmic exercises. Accordingly, you are well-advised to both study the 
code in StdRandom. java and StdStats. java on the booksite and to take advantage 
of these tried-and-true implementations. The easiest way to use these libraries (and to 
examine the code) is to download the source code from the booksite and put them in 
your working directory; various system-dependent mechanisms for using them with¬ 
out making multiple copies are also described on the booksite. 

Your own libraries. It is worthwhile to consider every program that you write as a li¬ 
brary implementation, for possible reuse in the future. 

■ Write code for the client, a top-level implementation that breaks the computa¬ 
tion up into manageable parts. 

■ Articulate an API for a library (or multiple APIs for multiple libraries) of static 
methods that can address each part. 

■ Develop an implementation of the API, with a mai n () that tests the methods 
independent of the client. 

Not only does this approach provide you with valuable software that you can later 
reuse, but also taking advantage of modular programming in this way is a key to suc¬ 
cessfully addressing a complex programming task. 
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intended result 


implementation 


random double public static double uniform(double a, double b) 

valueini a, b) { return a + StdRandom. randomQ * (b-a); } 


random int public static int uniform(int N) 

value in [0. . N) { return (int) (StdRandom. randomQ * N) ; } 


random i nt 
value in [1o. . hi ) 


public static int uniform(int lo, int hi) 

{ return lo + StdRandom.uniform(hi - lo); } 


random i nt value drawn 
from discrete distribution 
(i with probability a [ i ]) 


public static int discrete(doubl e[] a) 

{ // Entries in a[] must sum to 1. 

double r = StdRandom.random(); 
double sum = 0.0; 

for (int i = 0; i < a.length; i++) 

{ 

sum = sum + a[i ] ; 
if (sum >= r) return i; 

} 

return -1; 

} 


randomly shuffle the 
elements in an array of 
doubl e values 
(See Exercise 1.1.36 ) 


public static void shuffle(double[] a) 

{ 

int N = a.length; 

for (int i =0; i < N; i++) 

{ // Exchange a[i] with random element in a[i..N-l] 

int r = i + StdRandom.uniform(N-i); 
doubl e temp = a[i ] ; 
a[i] = a[r] ; 
a[r] = temp; 

} 

} 


Implementations of static methods in StdRandom library 
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the purpose of an api is to separate the client from the implementation: the client 
should know nothing about the implementation other than information given in the 
API, and the implementation should not take properties of any particular client into 
account. APIs enable us to separately develop code for various purposes, then reuse 
it widely. No Java library can contain all the methods that we might need for a given 
computation, so this ability is a crucial step in addressing complex programming ap¬ 
plications. Accordingly, programmers normally think of the API as a contract between 
the client and the implementation that is a clear specification of what each method is to 
do. Our goal when developing an implementation is to honor the terms of the contract. 
Often, there are many ways to do so, and separating client code from implementation 
code gives us the freedom to substitute new and improved implementations. In the 
study of algorithms, this ability is an important ingredient in our ability to understand 
the impact of algorithmic improvements that we develop. 
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Strings A String is a sequence of characters (char values). A literal String is a 
sequence of characters within double quotes, such as "Hel 1 o, Worl d". The data type 
Stri ng is a Java data type but it is not a primitive type. We consider Stri ng now be¬ 
cause it is a fundamental data type that almost every Java program uses. 

Concatenation. Java has a built-in concatenation operator (+) for String like the 
built-in operators that it has for primitive types, justifying the addition of the row in 
the table below to the primitive-type table on page 12. The result of concatenating two 
Stri ng values is a single Stri ng value, the first string followed by the second. 


type set of values 


Stri ng 


character 

sequences 


typical literals 


operators 


typical expressions 


expression 


value 


"AB" 

’Hello" 

H 2 ^ (concatenate) 
Java's String data type 


"Hi, " + "Bob" 
" 12 " + " 34 " 
" 1 " + „ + „ + „ 2 „ 


"Hi, Bob" 

"1234" 

" 1 + 2 " 


Conversion. Two primary uses of strings are to convert values that we can enter on a 
keyboard into data-type values and to convert data-type values to values that we can 
read on a display. Java has built-in operations for Stri ng to facilitate these operations. 
In particular, the language includes libraries Integer and Double that contain static 
methods to convert between Stri ng values and i nt values and between Stri ng values 
and doubl e values, respectively. 


public class Integer 

static int parseInt(Stri ng s) convert s to an i nt value 

static String toString(int i) convert i to a Stri ng value 

public class Double 

static double parseDouble(Stri ng s) convert s to a doubl e value 

static String toString (double x) convert x to a Stri ng value 

APIs for conversion between numbers and String values 
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Automatic conversion. We rarely explicitly use the static toStringO methods just 
described because Java has a built-in mechanism that allows us to convert from any data 
type value to a Stri ng value by using concatenation: if one of the arguments of + is a 
Stri ng, Java automatically converts the other argument to a Stri ng (if it is not already 
a Stri ng). Beyond usage like "The square root of 2.0 is " + Math. sqrt(2.0) 
this mechanism enables conversion of any data-type value to a Stri ng, by concatenat¬ 
ing it with the empty string " 

Command-line arguments. One important use of strings in Java programming is to 
enable a mechanism for passing information from the command line to the program. 
The mechanism is simple. When you type the j ava command followed by a library 
name followed by a sequence of strings, the Java system invokes the mai n () method in 
that library with an array of strings as argument: the strings typed after the library name. 
For example, the mai n () method in Bi narySearch takes one command-line argument, 
so the system creates an array of size one. The program uses that value, args[0], to 
name the file containing the whitelist, for use as the argument to In . readlntsC). An¬ 
other typical paradigm that we often use in our code is when a command-line argu¬ 
ment is intended to represent a number, so we use parselntO to convert to an i nt 
value or parseDoubl e() to convert to a doubl e value. 

computing with strings is an essential component of modern computing. For the 
moment, we make use of String just to convert between external representation of 
numbers as sequences of characters and internal representation of numeric data-type 
values. In section 1 . 2 , we will see that Java supports many, many more operations on 
Stri ng values that we use throughout the book; in section 1 . 4 , we will examine the 
internal representation of Stri ng values; and in chapter 5 , we consider in depth al¬ 
gorithms that process Stri ng data. These algorithms are among the most interesting, 
intricate, and impactful methods that we consider in this book. 
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Input and output The primary purpose of our standard libraries for input, out¬ 
put, and drawing is to support a simple model for Java programs to interact with the 
outside world. These libraries are built upon extensive capabilities that are available in 
Java libraries, but are generally much more complicated and much more difficult to 

learn and use. We begin by briefly reviewing the model. 

In our model, a Java program takes input values from 
command-line arguments or from an abstract stream of 
characters known as the standard input stream and writes 
to another abstract stream of characters known as the 
standard output stream. 

Necessarily, we need to consider the interface between 
Java and the operating system, so we need to briefly dis¬ 
cuss basic mechanisms that are provided by most modern 
operating systems and program-development environ¬ 
ments. You can find more details about your particular 
system on the booksite. By default, command-line argu¬ 
ments, standard input, and standard output are associated 
with an application supported by either the operating system or the program develop¬ 
ment environment that takes commands. We use the generic term terminal window to 
refer to the window maintained by this application, where we type and read text. Since 
early Unix systems in the 1970s this model has proven to be a convenient and direct way 
for us to interact with our programs and data. We add to the classical model a standard 
drawing that allows us to create visual representations for data analysis. 


standard input 


command-line 

arguments 



file I/O 


standard drawing 

A bird's-eye view of a Java program 


Commands and arguments. In the terminal window, we see a prompt, where we type 
commands to the operating system that may take arguments. We use only a few com¬ 
mands in this book, shown in the table below. Most often, we use the . j ava com¬ 
mand, to run our programs. As mentioned on page 35, Java classes have a mai n () static 
method that takes a Stri ng array args [] as its argument. That array is the sequence 
of command-line arguments that we type, provided to Java by the operating system. 

By convention, both Java and 
the operating system process 
the arguments as strings. If 
we intend for an argument to 
be a number, we use a method 
such as Integer. parselntO 
to convert it from String to 
the appropriate type. 


command 

arguments 

purpose 

j avac 

. j ava file name 

compile Java program 

java 

. cl ass file name (no extension) 
and command-line arguments 

run Java program 

more 

any text file name 

print file contents 


Typical operating-system commands 
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Standard output. Our StdOut library provides sup¬ 
port for standard output. By default, the system con¬ 
nects standard output to the terminal window. The 
print() method puts its argument on standard out¬ 
put; the printlnO method adds a newline; and the 
printfO method supports formatted output, as de¬ 
scribed next. Java provides a similar method in its 
System.out library; we use StdOut to treat standard 
input and standard output in a uniform manner (and 
to provide a few technical improvements). 

public class StdOut 

static void print(String s) prints 

static void println(String s) print s, followed by newline 

static void printlnO print a new line 

static void pri ntf (Stri ng f, ... ) formatted print 

Note: overloaded implementations are included for primitive types and for Object. 

API for our library of static methods for standard output 

To use these methods, download into 
your working directory StdOut. java 
from the booksite and use code such as 
StdOut.println("Hello, World"); 
to call them. A sample client is shown 
at right. 

Formatted output. In its simplest 
form, printfO takes two arguments. 

The first argument is a format string 
that describes how the second argu¬ 
ment is to be converted to a string for 
output. The simplest type of format 
string begins with % and ends with a 
one-letter conversion code. The conversion codes that we 
use most frequently are d (for decimal values from Java’s 
integer types), f (for floating-point values), and s (for 
Stri ng values). Between the % and the conversion code 
is an integer value that specifies the field width of the 


% java RandomSeq 5 100.0 200.0 

123.43 

153.13 

144.38 

155.18 

104.02 


public class RandomSeq 
{ 

public static void main (Stri ng[] args) 

{ // Print N random values in (lo, hi). 

int N = Integer.parselnt(args[0]); 
double lo = Double.parseDouble(args[1]); 
double hi = Double.parseDouble(args[2]); 
for (int i =0; i < N; i++) 

{ 

double x = StdRandom.uniform(lo, hi); 
StdOut.printf("%.2f\n", x); 

} 

} 

} 


Sample StdOut client 


prompt 


call the static method 
mainQ in RandomSeq 


V 


% java RandomSeq 5 100.0 200.0 


invoke 

Java 

runtime 


args[0] 

args[l] 


args[2] 


Anatomy of a command 
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converted value (the number of characters in the converted output string). By default, 
blank spaces are added on the left to make the length of the converted output equal to 
the field width; if we want the spaces on the right, we can insert a minus sign before the 
field width. (If the converted output string is bigger than the field width, the field width 
is ignored.) Following the width, we have the option of including a period followed by 
the number of digits to put after the decimal point (the precision) for a doubl e value 
or the number of characters to take from the beginning of the string for a St ri ng value. 
The most important thing to remember about using printfO is that the conversion 
code in the format and the type of the corresponding argument must match. That is, Java 
must be able to convert from the type of the argument to the type required by the con¬ 
version code. The first argument of pri ntf () is a Stri ng that may contain characters 
other than a format string. Any part of the argument that is not part of a format string 
passes through to the output, with the format string replaced by the argument value 
(converted to a Stri ng as specified). For example, the statement 

StdOut.printf("PI is approximately %.2f\n", Math.PI); 
prints the line 

PI is approximately 3.14 

Note that we need to explicitly include the newline character \n in the argument in 
order to print a new line with pri ntf (). The pri ntf () function can take more than 
two arguments. In this case, the format string will have a format specifier for each ad¬ 
ditional argument, perhaps separated by other characters to pass through to the out¬ 
put. You can also use the static method Stri ng. format () with arguments exactly as 
just described for printfO to get a formatted string without printing it. Formatted 
printing is a convenient mechanism that allows us to develop compact code that can 
produce tabulated experimental data (our primary use in this book). 


type 

code 

typical 

literal 

int 

d 

512 

double 

f 

1595.1680010754388 


e 


Stri ng 

s 

"Hello, World" 


sample converted string 

format strings values for output 


"%14d" 

" 512" 

"%-14d" 

"512 

"%14.2f" 

" 1595.17" 

"%.7f" 

"1595.1680011" 

"%14.4e" 

" 1.5952e+03" 

"%14s" 

" Hello, World" 

"%-14s" 

"Hello, World 

"%-14.5s" 

"Hello 


Format conventions for printfO (see the booksite for many other options) 
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Standard input. Our Stdln library 
takes data from the standard input 
stream that may be empty or may 
contain a sequence of values sepa¬ 
rated by whitespace (spaces, tabs, 
newline characters, and the like). By 
default, the system connects stan¬ 
dard output to the terminal win¬ 
dow—what you type is the input 
stream (terminated by <ctrl-d> or 
<ctrl -z>, depending on your termi¬ 
nal window application). Each value 
is a String or a value from one of 
Java’s primitive types. One of the key 
features of the standard input stream 
is that your program consumes values when it reads them. Once 
your program has read a value, it cannot back up and read it again. 
This assumption is restrictive, but it reflects physical characteristics 
of some input devices and simplifies implementing the abstrac¬ 
tion. Within the input stream model, the static methods in this li¬ 
brary are largely self-documenting (described by their signatures). 

public class Stdln 


static 

boolean 

i sEmptyO 

true if no more values, fal se otherwise 

static 

i nt 

readlntO 

read a value of type int 

static 

double 

readDouble() 

read a value of type doubl e 

static 

float 

readFloatO 

read a value of type float 

static 

long 

readLongO 

read a value of type 1 ong 

static 

boolean 

readBoolean() 

read a value of type bool ean 

static 

char 

readCharO 

read a value of type char 

static 

byte 

readByte() 

read a value of type byte 

static 

Stri ng 

readStri ng() 

read a value of type String 

static 

boolean 

hasNextLine() 

is there another line in the input stream? 

static 

Stri ng 

readLi ne() 

read the rest of the line 

static 

Stri ng 

readAll() 

read the rest of the input stream 


API for our library of static methods for standard input 


% java Average 

1.23456 

2.34567 

3.45678 

4.56789 

cctrl -d> 

Average is 2.90123 


public class Average 
{ 

public static void main(String[] args) 

{ // Average the numbers on Stdln. 

double sum = 0.0; 
int cnt = 0; 

while (! Stdln. isEmptyO) 

{ // Read a number and cumulate the sum. 

sum += Stdln.readDouble(); 
cnt++; 

} 

double avg = sum / cnt; 

StdOut.printf("Average is %.5f\n", avg); 

} 

} 

Sample Stdln client 
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Redirection and piping. Standard input and output enable us to take advantage of 
command-line extensions supported by many operating-systems. By adding a simple 
directive to the command that invokes a program, we can redirect its standard output 
to a file, either for permanent storage or for input to another program at a later time: 

% java RandomSeq 1000 100.0 200.0 > data.txt 

This command specifies that the standard output stream is not to be printed in the ter¬ 
minal window, but instead is to be written to a text file named data. txt. Each call to 

StdOut.print()orStdOut.println() 
appends text at the end of that file. In 
this example, the end result is a file that 
contains 1,000 random values. No out¬ 
put appears in the terminal window: it 
goes directly into the file named after 
the > symbol. Thus, we can save away 
information for later retrieval. Not that 
we do not have to change RandomSeq in 
any way—it is using the standard out¬ 
put abstraction and is unaffected by our 
use of a different implementation of 
that abstraction. Similarly, we can redi¬ 
rect standard input so that Stdln reads 
data from a file instead of the terminal 
application: 

% java Average < data.txt 

This command reads a sequence of 
numbers from the file data.txt and 
computes their average value. Specifi¬ 
cally, the < symbol is a directive that tells 
the operating system to implement the 
standard input stream by reading from 
the text file data. txt instead of waiting 
for the user to type something into the 
terminal window. When the program calls Stdln. readDoubl e(), the operating system 
reads the value from the file. Combining these to redirect the output of one program to 
the input of another is known as piping: 

% java RandomSeq 1000 100.0 200.0 | java Average 


redirecting from a file to standard input 

% java Average < data.txt 


data.txt 



redirecting standard output to a file 

% java RandomSeq 1000 100.0 200.0 > data.txt 



piping the output of one program to the input of another 

% java RandomSeq 1000 100.0 200.0 | java Average 



Redirection and piping from the command line 
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This command specifies that standard output for RandomSeq and standard input for 
Average are the same stream. The effect is as if RandomSeq were typing the numbers it 
generates into the terminal window while Average is running. This difference is pro¬ 
found, because it removes the limitation on the size of the input and output streams that 
we can process. For example, we could replace 1000 in our example with 1000000000, 
even though we might not have the space to save a billion numbers on our computer 
(we do need the time to process them). When RandomSeq calls StdOut.printlnO, a 
string is added to the end of the stream; when Average calls Stdln.readlntO, a string 
is removed from the beginning of the stream. The timing of precisely what happens is 
up to the operating system: it might run RandomSeq until it produces some output, and 
then run Average to consume that output, or it might run Average until it needs some 
output, and then run RandomSeq until it produces the needed output. The end result 
is the same, but our programs are freed from worrying about such details because they 
work solely with the standard input and standard output abstractions. 

Input and output from a file. Our In and Out libraries provide static methods that 
implement the abstraction of reading from and writing to a file the contents of an ar¬ 
ray of values of a primitive type (or String). We use readlnts(), readDoubles(), 
and readStringsO in the In library and writelntsO, writeDoublesO, and 
wri teStri ngs O in the Out library. The named argument can be a file or a web page. 
For example, this ability allows us to use a file and standard input for two different pur¬ 
poses in the same program, as in Bi narySearch. The In and Out libraries also imple¬ 
ment data types with instance methods that allow us the more general ability to treat 
multiple files as input and output streams, and web pages as input streams, so we will 
revisit them in section 1 . 2 . 

public class In 

static int[] readInts(String name) read int values 

static doublet] readDoubl es (St ri ng name) read doubl e values 

static String [] readStri ngs(Stri ng name) read St ri ng values 

public class Out 


static void write(int[] a, String name) 
static void write(double[] a, String name) 
static void wri te(Stri ng [] a, String name) 


write i nt values 
write doubl e values 
write Stri ng values 


Note 1: Other primitive types are supported. 

Note 2: Stdln and StdOut are supported (omit name argument). 


APIs for our static methods for reading and writing arrays 
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Standard drawing (basic methods). Up to this point, 
our input/output abstractions have focused exclusively 
on text strings. Now we introduce an abstraction for 
producing drawings as output. This library is easy to 
use and allows us to take advantage of a visual medi¬ 
um to cope with far more information than is possible 
with just text. As with standard input/output, our stan¬ 
dard drawing abstraction is implemented in a library 
StdDraw that you can access by downloading the file 
StdDraw. java from the booksite into your working 
directory. Standard draw is very simple: we imagine an 
abstract drawing device capable of drawing lines and 
points on a two-dimensional canvas. The device is ca¬ 
pable of responding to the commands to draw basic 
geometric shapes that our programs issue in the form 
of calls to static methods in StdDraw, including meth¬ 
ods for drawing lines, points, text strings, circles, rect¬ 
angles, and polygons. Like the methods for standard 
input and standard output, these methods are nearly 
self-documenting: StdDraw.1ine() draws a straight 
line segment connecting the point (x 0 , y 0 ) with the 
point (xj ,y,) whose coordinates are given as arguments. 
StdDraw. point() draws a spot centered on the point 
(x, y ) whose coordinates are given as arguments, and so 
forth, as illustrated in the diagrams at right. Geometric 
shapes can be filled (in black, by default). The default 
scale is the unit square (all coordinates are between 0 
and 1). The standard implementation displays the can¬ 
vas in a window on your computer’s screen, with black 
lines and points on a white background. 


StdDraw.point(xO, yO); 
StdDraw.Iine(x0, yO, xl, yl); 



StdDraw.cirete(x, y, r); 



StdDraw.square(x, y, r); 



doublet] x = {xO, xl, x2, xS}; 
doublet] y = {yO, yl, y2, y3}; 
StdDraw.polygon(x, y); 



www.it-ebooks.info 


























1.1 ■ Basic Programming Model 


43 


public class StdDraw 


static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 

static 

voi d 


line(double xO, double yO, double xl, double yl) 
point(double x, double y) 
text(double x, double y, String s) 
ci rcle(double x, double y, double r) 
filledCi rcle(double x, double y, double r) 
el 1ipse(double x, double y, double rw, double rh) 
filledEllipse(double x, double y, double rw, double rh) 
square(double x, double y, double r) 
filledSquare(double x, double y, double r) 
rectangle(double x, double y, double rw, double rh) 
filledRectangle(double x, double y, double rw, double rh) 
polygon(double[] x, doublet] y) 
filledPolygon(double[] x, doublet] y) 


API for our library of static methods for standard drawing (drawing methods) 


Standard drawing (control methods). The library also includes methods to change 
the scale and size of the canvas, the color and width of the lines, the text font, and 
the timing of drawing (for use in animation). As arguments for setPenColorO you 
can use one of the predefined colors BLACK, BLUE, CYAN, DARK_GRAY, CRAY, GREEN, 
LIGHT_GRAY, MAGENTA, ORANGE, PINK, RED, BOOK_RED, WHITE, and YELLOW that are de¬ 
fined as constants in StdDraw (so we refer to one of them with code like StdDraw. RED). 
The window also includes a menu option to save your drawing to a file, in a format 
suitable for publishing on the web. 


public class StdDraw 


stati c 

voi d 

setXscale(double xO, 

double xl) 

reset x range to (x 0 , xf 

stati c 

voi d 

setYscale(double yO, 

double yl) 

reset y range to (y 0 , yf 

static 

voi d 

setPenRadius(double 

r) 

set pen radius to r 

static 

voi d 

setPenColor(Color c) 


set pen color to c 

stati c 

voi d 

setFont(Font f) 


set text font tof 

static 

voi d 

setCanvasSize(i nt w, 

int h) 

set canvas to w-by-h window 

static 

voi d 

clear(Color c) 


clear the canvas; color it c 

static 

voi d 

show(int dt) 


show all; pause dt milliseconds 


API for our library of static methods for standard drawing (control methods) 
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in this book, we use StdDraw for data analysis and for creating visual representations 
of algorithms in operation. The table at on the opposite page indicates some possibli- 
ties; we will consider many more examples in the text and the exercises throughout the 
book. The library also supports animation —of course, this topic is treated primarily on 
the booksite. 
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data 


function 

values 


array of 
random 
values 


sorted array 
of random 
values 


plot implementation (code fragment) 


result 


int N = 100; 

StdDraw.setXscale(0, N); 
StdDraw.setYscale(0, N*N); 
StdDraw.setPenRadius(.Ol); 
for (int i = 1; i <= N; i++) 

{ 

StdDraw.point(i, i); 

StdDraw. poi nt(i , i*i); 

StdDraw. point(i , i*Math.log(i)) ; 



int N = 50; 

doublet] a = new double[N]; 
for (int i =0; i < N; i++) 
a[i] = StdRandom.random(); 
for (int i =0; i < N; i++) 

{ 

double x = 1.0*i/N; 
double y = a[i]/2.0; 
double rw = 0.5/N; 
double rh = a[i]/2.0; 

StdDraw.filledRectangle(x, y, rw, rh); 

} 



int N = 50; 

doublet] a = new doubletN]; 
for (int i =0; i < N; i++) 
a[i] = StdRandom.random(); 

Arrays. sort(a) ; 

for (int i =0; i < N; i++) 

{ 

double x = 1.0*i/N; 
double y = a[i]/2.0; 
double rw = 0.5/N; 
double rh = a[i]/2.0; 

StdDraw.filledRectangle(x, y, rw, rh); 

} 



StdDraw plotting examples 
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Binary search The sample Java program that we started with, shown on the facing 
page, is based on the famous, effective, and widely used binary search algorithm. This 
example is a prototype of the way in which we will examine new algorithms throughout 
the book. As with all of the programs we consider, it is both a precise definition of the 
method and a complete Java implementation that you can download from the booksite. 


successful search for 23 


Binary search. We will study the binary search algorithm in detail in section 3 . 2 , 
but a brief description is appropriate here. The algorithm is implemented in the static 

method rank(), which takes an integer key and 
a sorted array of i nt values as arguments and re¬ 
turns the index of the key if it is present in the 
array, -1 otherwise. It accomplishes this task by 
maintaining variables lo and hi such that the key 
is in a [1 o. . hi ] if it is in the array, then entering 
into a loop that tests the middle entry in the in¬ 
terval (at index mi d). If the key is equal to a [mi d], 
the return value is mi d; otherwise the method cuts 
the interval size about in half, looking at the left 
half if the key is less than a [mi d] and at the right 
half if the key is greater than a [mi d] . The process 
terminates when the key is found or the interval is 
empty. Binary search is effective because it needs 
to examine just a few ar¬ 
ray entries (relative to the 
size of the array) to find 
the key (or determine that 

Binary search in an ordered array it is not there). 
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tinyW.txt tinyT.txt 


Development client. For every algorithm implementation, 
we include a development client mai n () that you can use with 
sample input files provided in the book and on the booksite 
to learn about the algorithm and to test its performance. In 
this example, the client reads integers from the file named on 
the command line, then prints any integers on standard input 
that do not appear in the file. We use small test files such as 
those shown at right to demonstrate this behavior, and as the 
basis for traces and examples such as those at left above. We 
use large test files to model real-world applications and to test 
performance (see page 48). 


84 

23 

48 

50 

68 

10> 

10 

99 

18 

18 

98 

23 

12 

98 

23 

84 

54 

11 

57 

10 

48 

48 

33 

77) 

16 

13 

77 

54 

11 

98 

29 

77 


77 


68 


> not in 

tinyW.txt 


Small test files for 
BinarySearch test client 
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Binary Search 


import java.util.Arrays; 

public class BinarySearch 

{ 

public static int rank(int key, int[] a) 

{ // Array must be sorted. 

int lo =0; 

int hi = a.length - 1; 

while (lo <= hi) 

{ // Key is in a[lo..hi] or not present. 

int mid = lo + (hi - lo) / 2; 
if (key < a[mid]) hi = mid - 1; 

else if (key > a[mid]) lo = mid + 1; 
else return mid; 


} 

return 


} 


-l; 


public static void main(String[] args) 

{ 

int[] whitelist = In.readlnts(args[0]); 
Arrays.sort(whitel i st); 

while (!Stdln.isEmptyO) 

{ // Read key, print if not in whitelist. 

int key = Stdln.readlnt(); 
if (rank(key, whitelist) < 0) 

StdOut.println(key); 

} 


This program takes the name of a whitelist file (a sequence of integers) as argument and filters any 
entry that is on the whitelist from standard input, leaving only integers that are not on the whitelist 
on standard output. It uses the binary search algorithm, implemented in the static method rank(), 
to accomplish the task efficiently. See sec¬ 
tion 3.1 for a full discussion of the binary 
search algorithm, its correctness, its per¬ 
formance analysis, and its applications. 


% java BinarySearch tinyW.txt < tinyT.txt 

50 

99 

13 
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Whitelisting. When possible, our development clients are intended to mirror practical 
situations and demonstrate the need for the algorithm at hand. In this case, the process 
is known as whitelisting. Specifically, imagine a credit card company that needs to check 
whether customer transactions are for a valid account. To do so, it can 

■ Keep customers account numbers in a file, which we refer to as a whitelist. 

■ Produce the account number associated with each transaction in the standard 
input stream. 

■ Use the test client to put onto standard output the numbers that are not associat¬ 
ed with any customer. Presumably the company would refuse such transactions. 

It would not be unusual for a big company with millions of customers to have to pro¬ 
cess millions of transactions or more. To model this situation, we provide on the book- 
site the files 1 argeW.txt (1 million integers) and 1 argeT.txt (10 million integers). 

Performance. A working program is often not sufficient. For example, a much simpler 
implementation of rank(), which does not even require the array to be sorted, is to 
check every entry, as follows: 

public static int rank(int key, int[] a) 

{ 

for (int i =0; i < a.length; i++) 
if (a[i] == key) return i; 
return -1; 

} 

Given this simple and easy-to-understand solution, why do we use mergesort and bi¬ 
nary search? If you work exercise 1 . 1 . 38 , you will see that your computer is too slow 
to run this brute-force implementation of rank() for large numbers of inputs (say, 1 
million whitelist entries and 10 million transactions). Solving the whitelist problem for 
a large number of inputs is not feasible without efficient algorithms such as binary search 
and mergesort. Good performance is often of critical importance, so we lay the ground¬ 
work for studying performance in section 1.4 and analyze the performance character¬ 
istics of all of our algorithms (including binary search, in section 3.1 and mergesort, 
in section 2 . 2 ). 

in the present context, our goal in thoroughly outlining our programming model 
is to ensure that you can run code like Bi narySearch on your computer, use it on test 
data like ours, and modify it to adapt to various situations (such as those described in 
the exercises at the end of this section), in order to best understand its applicability. 
The programming model that we have sketched is designed to facilitate such activities, 
which are crucial to our approach to studying algorithms. 


www.it-ebooks.info 


1.1 ■ Basic Programming Model 


49 


1argeW.txt 

489910 

18940 

774392 

490636 

125544 

407391 

115771 

992663 

923282 

176914 

217904 

571222 

519039 

395667 

I 

1,000,000 

i nt values 


largeT.txt 


944443 

293674 

572153 

600579 

499569 

984875 

763178 

295754 

44696 

207807 

138910 

903531 

140925 

699418 

759984 

199694 

774549 

635871 

161828 

805380 



not in 

largeW.txt 


l 

10,000,000 

i nt values 


% java BinarySearch largeW.txt < largeT.txt 

499569 

984875 

295754 

207807 

140925 

161828 

3,675,966 

int values 


Large files for BinarySearch test client 
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Perspective In this section, we have described a fine and complete programming 
model that served (and still serves) many programmers for many decades. Modern 
programming, however, goes one step further. This next level is called data abstraction, 
sometimes known as object-oriented programming, and is the subject of the next sec¬ 
tion. Simply put, the idea behind data abstraction is to allow a program to define data 
types (sets of values and sets of operations on those values), not just static methods that 
operate on predefined data types. 

Object-oriented programming has come into widespread use in recent decades, and 
data abstraction is central to modern program development. We embrace data abstrac¬ 
tion in this book for three primary reasons: 

■ It enables us to expand our ability to reuse code through modular programming. 
For example, our sorts in chapter 2 and binary search and other algorithms in 
chapter 3 allow clients to make use of the same code for any type of data (not 
just integers), including one defined by the client. 

■ It provides a convenient mechanism for building so-called linked data structures 
that provide more flexibility than arrays and are the basis of efficient algorithms 
in many settings. 

■ It enables us to precisely define the algorithmic challenges that we face. For ex¬ 
ample, our union-find algorithms in section 1 . 5 , our priority-queue algorithms 
in section 2 . 4 , and our symbol-table algorithms in chapter 3 are all oriented 
toward defining data structures that enable efficient implementations of a set of 
operations. This challenge aligns perfectly with data abstraction. 

Despite all of these considerations, our focus remains on the study of algorithms. In 
this context, we proceed to consider next the essential features of object-oriented pro¬ 
gramming that are relevant to our mission. 
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| Q&A 


Q. What is Java bytecode? 

A. A low-level version of your program that runs on the Java virtual machine. This level 
of abstraction makes it easier for the developers of Java to ensure that our programs run 
on a broad variety of devices. 

Q. It seems wrong that Java should just let i nts overflow and give bad values. Shouldn’t 
Java automatically check for overflow? 

A. This issue is a contentious one among programmers. The short answer is that the 
lack of such checking is one reason such types are called primitive data types. A little 
knowledge can go a long way in avoiding such problems. We use the i nt type for small 
numbers (less than ten decimal digits), and the 1 ong type when values run into the bil¬ 
lions or more. 

Q. What is the value of Math. abs (-2147483648)? 

A. -2147483648. This strange (but true) result is a typical example of the effects of 
integer overflow. 

Q. How can I initialize a doubl e variable to infinity? 

A. Java has built-in constants available for this purpose: Doubl e. POSITIVE_INFINITY 
and Doubl e.NEGATIVE_INFINITY. 

Q. Can you compare a doubl e to an i nt? 

A. Not without doing a type conversion, but remember that Java usually does the req¬ 
uisite type conversion automatically. For example, if x is an i nt with the value 3, then 
the expression (x < 3.1) is true—Java converts x to doubl e (because 3.1 is a doubl e 
literal) before performing the comparison. 

Q. What happens if I use a variable before initializing it to a value? 

A. Java will report a compile-time error if there is any path through your code that 
would lead to use of an uninitialized variable. 

Q. What are the values of 1/0 and 1.0/0.0 as Java expressions? 

A. The first generates a runtime exception for division by zero (which stops your pro¬ 
gram because the value is undefined); the second has the value Infi ni ty. 
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Q&A (continued) 


Q. Can you use < and > to compare Stri ng variables? 

A. No. Those operators are defined only for primitive types. See page 80. 

Q. What is the result of division and remainder for negative integers? 

A. The quotient a/b rounds toward 0; the remainder a % b is defined such that (a / 
b) * b + a % b is always equal to a. For example, -14/3 and 14/-3 are both -4, but 
-14 % 3 is -2 and 14 % -3 is 2. 

Q. Why do we say (a && b) and not (a & b)? 

A. The operators &, |, and A are bitwise logical operations for integer types that do and, 
or, and exclusive or (respectively) on each bit position. Thus the value of 10&6 is 14 and 
the value of 10A6 is 12. We use these operators rarely (but occasionally) in this book. 
The operators && and | | are valid only in boolean expressions are included separately 
because of short-circuiting: an expression is evaluated left-to-right and the evaluation 
stops when the value is known. 

Q. Is ambiguity in nested i f statements a problem? 

A. Yes. In lava, when you write 

if <exprl> if <expr2> <stmntA> else <stmntB> 

it is equivalent to 

if <exprl> {if <expr2> <stmntA> else <stmntB> } 
even if you might have been thinking 

if <exprl> {if <expr2> <stmntA> } else <stmntB> 

Using explicit braces is a good way to avoid this dangling else pitfall. 

Q. What is the difference between a for loop and its whi 1 e formulation? 

A. The code in the for loop header is considered to be in the same block as the for 
loop body. In a typical for loop, the incrementing variable is not available for use in 
later statements; in the corresponding whi 1 e loop, it is. This distinction is often a rea¬ 
son to use a whi 1 e instead of a for loop. 

Q. Some Java programmers use i nt a [] instead of i nt[] a to declare arrays. What’s 
the difference? 
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A. In Java, both are legal and equivalent. The former is how arrays are declared in C. 
The latter is the preferred style in Java since the type of the variable i nt [] more clearly 
indicates that it is an array of integers. 

Q. Why do array indices start at 0 instead of 1? 

A. This convention originated with machine-language programming, where the ad¬ 
dress of an array element would be computed by adding the index to the address of the 
beginning of an array. Starting indices at 1 would entail either a waste of space at the 
beginning of the array or a waste of time to subtract the 1. 

Q. If a [] is an array, why does StdOut. pri ntl n(a) print out a hexadecimal integer, 
such as @f62373 , instead of the elements of the array? 

A. Good question. It is printing out the memory address of the array, which, unfortu¬ 
nately, is rarely what you want. 

Q. Why are we not using the standard Java libraries for input and graphics? 

A. We are using them, but we prefer to work with simpler abstract models. The Java 
libraries behind Stdln and StdDraw are built for production programming, and the 
libraries and their APIs are a bit unwieldy. To get an idea of what they are like, look at 
the code in Stdln .java and StdDraw. java. 

Q. Can my program reread data from standard input? 

A. No. You only get one shot at it, in the same way that you cannot undo pri ntl n (). 

Q. What happens if my program attempts to read after standard input is exhausted? 

A. You will get an error. Stdln . i sEmpty () allows you to avoid such an error by check¬ 
ing whether there is more input available. 

Q. What does this error message mean? 

Exception in thread "main" java.lang.NoClassDefFoundError: Stdln 

A. You probably forgot to put Stdln. java in your working directory. 

Q. Can a static method take another static method as an argument in Java? 

A. No. Good question, since many other languages do support this capability. 
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EXERCISES 

1 .1.1 Give the value of each of the following expressions: 

a. ( 0 + 15 ) / 2 

b. 2.Oe-6 * 100000000.1 

c. true && false | | true && true 

1.1.2 Give the type and value of each of the following expressions: 

a. (1 + 2.236)/2 

b. 1 + 2 + 3 + 4.0 

c. 4.1 >= 4 

d. 1 + 2 + "3" 

1.1.3 Write a program that takes three integer command-line arguments and prints 
equal if all three are equal, and not equal otherwise. 

1.1.4 What (if anything) is wrong with each of the following statements? 

a. if (a > b) then c = 0; 

b. if a > b { c = 0; } 

c. if (a > b) c = 0; 

d. if (a > b) c = 0 else b = 0; 

1.1.5 Write a code fragment that prints true if the doubl e variables x and y are both 
strictly between 0 and 1 and f al se otherwise. 

1.1.6 What does the following program print? 

int f = 0; 
int g = 1; 

for (int i =0; i <=15; i++) 

{ 

StdOut.println(f); 
f = f + g; 

g = f - g; 

} 
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1.1.7 Give the value printed by each of the following code fragments: 


a. double t = 9.0; 

while (Math.abs(t - 9.0/t) > 
t = (9.0/t + t) / 2.0; 
StdOut.printf("%.5f\n", t); 


. 001 ) 


b. int sum = 0; 

for (int i =1; i < 1000; i++) 
for (int j = 0; j < i; j++) 
sum++; 

StdOut.println(sum); 


c. int sum = 0; 

for (int i =1; i < 1000; i *= 2) 
for (int j = 0; j < N; j++) 
sum++; 

StdOut.println(sum); 


1.1.8 What do each of the following print? 

a. System.out.println('b'); 

b. System.out.println('b' + 'c'); 

c. System.out.println((char) ('a' + 4)); 

Explain each outcome. 

1 . 1.9 Write a code fragment that puts the binary representation of a positive integer N 
into a String s. 

Solution: Java has a built-in method Integer. toBi naryStri ng(N) for this job, but 
the point of the exercise is to see how such a method might be implemented. Here is a 
particularly concise solution: 

String s = 

for (int n = N; n > 0; n /= 2) 
s = (n % 2) + s; 
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EXERCISES (continued) 


1.1.10 What is wrong with the following code fragment? 
int[] a; 

for (int i =0; i <10; i++) 
a[i] = i * i ; 

Solution: It does not allocate memory for a[] with new. This code results in a 
variable a might not have been initialized compile-time error. 

1 . 1.11 Write a code fragment that prints the contents of a two-dimensional boolean 
array, using * to represent true and a space to represent fal se. Include row and column 
numbers. 

1.1.12 What does the following code fragment print? 

int[] a = new int[10]; 
for (int i =0; i <10; i++) 
a[i] = 9 - i; 

for (int i =0; i <10; i++) 
a[i] = a[a[i]]; 
for (int i =0; i <10; i++) 

System.out.println(i); 

1.1.13 Write a code fragment to print the transposition (rows and columns changed) 
of a two-dimensional array with M rows and N columns. 

1.1.14 Write a static method lg() that takes an int value N as argument and returns 
the largest i nt not larger than the base-2 logarithm of N. Do not use Math. 

1.1.15 Write a static method histogramO that takes an array a[] of i nt values and 
an integer M as arguments and returns an array of length M whose i th entry is the num¬ 
ber of times the integer i appeared in the argument array. If the values in a[] are all 
between 0 and M-l, the sum of the values in the returned array should be equal to 
a.length. 

1.1.16 Give the value of exRl (6): 

public static String exRlfint n) 

{ 

if (n <= 0) return 

return exRl(n-S) + n + exRl(n-2) + n; 

} 
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1.1.17 Criticize the following recursive function: 

public static String exR2(int n) 

{ 

String s = exR2(n-3) + n + exR2(n-2) + n; 
if (n <= 0) return 
return s; 

} 

Answer: The base case will never be reached. A call to exR2(3) will result in calls to 
exR2(0), exR2(-3), exR3(-6), and so forth until a StackOverflowError occurs. 

1.1.18 Consider the following recursive function: 

public static int mystery(int a, int b) 

{ 

if (b == 0) return 0; 

if (b % 2 == 0) return mystery(a+a, b/2); 

return mystery(a+a, b/2) + a; 

} 

What are the values of mystery(2, 25) and mystery(3, 11)? Given positive integers 
a and b, describe what value mystery (a , b) computes. Answer the same question, but 
replace + with * and replace return 0 with return 1. 

1.1.19 Run the following program on your computer: 

public class Fibonacci 

{ 

public static long F(int N) 

{ 

if (N == 0) return 0; 
if (N == 1) return 1; 
return F(N-l) + F(N-2); 

} 

public static void main(String[] args) 

{ 

for (int N = 0; N < 100; N++) 

StdOut.println(N + " " + F(N)); 

} 

} 
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EXERCISES (continued) 


What is the largest value of N for which this program takes less 1 hour to compute the 
value of F(N)? Develop a better implementation of F(N) that saves computed values in 
an array. 

1 . 1.20 Write a recursive static method that computes the value of In (N\) 

1 .1.21 Write a program that reads in lines from standard input with each line contain¬ 
ing a name and two integers and then uses pri ntf () to print a table with a column of 
the names, the integers, and the result of dividing the first by the second, accurate to 
three decimal places. You could use a program like this to tabulate batting averages for 
baseball players or grades for students. 

1 . 1.22 Write a version of Bi narySearch that uses the recursive rankf) given on page 
25 and traces the method calls. Each time the recursive method is called, print the argu¬ 
ment values 1 o and hi , indented by the depth of the recursion. Hint: Add an argument 
to the recursive method that keeps track of the depth. 

1 . 1.23 Add to the Bi narySearch test client the ability to respond to a second argu¬ 
ment: + to print numbers from standard input that are not in the whitelist, - to print 
numbers that are in the whitelist. 

1 . 1.24 Give the sequence of values of p and q that are computed when Euclid’s algo¬ 
rithm is used to compute the greatest common divisor of 105 and 24. Extend the code 
given on page 4 to develop a program Euclid that takes two integers from the command 
line and computes their greatest common divisor, printing out the two arguments for 
each call on the recursive method. Use your program to compute the greatest common 
divisor or 1111111 and 1234567. 

1 . 1.25 Use mathematical induction to prove that Euclid’s algorithm computes the 
greatest common divisor of any pair of nonnegative integers p and q. 
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| CREATIVE PROBLEMS 


1 . 1.26 Sorting three numbers. Suppose that the variables a, b, c, and t are all of the 
same numeric primitive type. Show that the following code puts a, b, and c in ascending 
order: 


if 

(a > 

b) 

{ 

t = a; 

a = b; 

b = t; } 

if 

(a > 

c) 

{ 

t = a; 

a = c; 

c = t; } 

if 

(b > 

c) 

{ 

t = b; 

b = c; 

c = t; } 


1 . 1.27 Binomial distribution. Estimate the number of recursive calls that would be 
used by the code 

public static double binomia!(int N, int k, double p) 

{ 

if ((N == 0) || (k < 0)) return 1.0; 

return (1.0 - p)*binomial(N-l, k) + p*binomial(N-l, k-1); 

} 

to compute binomial (100, 50). Develop a better implementation that is based on 
saving computed values in an array. 

1 . 1.28 Remove duplicates. Modify the test client in Bi narySearch to remove any du¬ 
plicate keys in the whitelist after the sort. 

1 . 1.29 Equal keys. Add to Bi narySearch a static method rank() that takes a key and 
a sorted array of i nt values (some of which may be equal) as arguments and returns the 
number of elements that are smaller than the key and a similar method count () that 
returns the number of elements equal to the key. Note: If i and j are the values returned 
by rank(key, a) and count(key, a) respectively, then a [i. .i+j-1 ] are the values in 
the array that are equal to key. 

1 . 1.30 Array exercise. Write a code fragment that creates an iV-by-iV boolean array 
a [] [] such that a [i ] [ j ] is true if i and j are relatively prime (have no common fac¬ 
tors), and fal se otherwise. 

1 . 1.31 Random connections. Write a program that takes as command-line arguments 
an integer N and a doubl e value p (between 0 and 1), plots N equally spaced dots of size 
.05 on the circumference of a circle, and then, with probability p for each pair of points, 
draws a gray line connecting them. 
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CREATIVE PROBLEMS (continued) 


1 . 1.32 Histogram. Suppose that the standard input stream is a sequence of double 
values. Write a program that takes an integer N and two doubl e values / and r from the 
command line and uses StdDraw to plot a histogram of the count of the numbers in the 
standard input stream that fall in each of the N intervals defined by dividing (l, r) into 
N equal-sized intervals. 

1 . 1.33 Matrix library. Write a library Mat ri x that implements the following API: 


public class Matrix 


static 

double 

static 

doublet] [] 

static 

doublet] [] 

static 

double □ 

static 

double □ 


dot(double[] x, doublet] y) 
mult(double[] [] a, doublet] [] b) 
transpose(double[] t] a) 
mult(double[] [] a, doublet] x) 
mult(double[] y, doublet] [] a) 


vector dot product 
matrix-matrix product 
transpose 

matrix-vector product 
vector-matrix product 


Develop a test client that reads values from standard input and tests all the methods. 

1 . 1.34 Filtering. Which of the following require saving all the values from standard 
input (in an array, say), and which could be implemented as a filter using only a fixed 
number of variables and arrays of fixed size (not dependent on N) ? . For each, the input 
comes from standard input and consists of N real numbers between 0 and 1. 

■ Print the maximum and minimum numbers. 

■ Print the median of the numbers. 

■ Print the k th smallest value, for k less than 100. 

■ Print the sum of the squares of the numbers. 

■ Print the average of the N numbers. 

■ Print the percentage of numbers greater than the average. 

■ Print the N numbers in increasing order. 

■ Print the N numbers in random order. 
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EXPERIMENTS 

1.1.35 Dice simulation. The following code computes the exact probability distribu¬ 
tion for the sum of two dice: 

int SIDES = 6; 

doublet] dist = new double[2*SIDES+1]; 
for (int i =1; i <= SIDES; i++) 
for (int j = 1; j <= SIDES; j++) 
dist[i+j] += 1.0; 

for (int k = 2; k <= 2*SIDES; k++) 
dist[k] /= 36.0; 

The value di st [i ] is the probability that the dice sum to k. Run experiments to vali¬ 
date this calculation simulating N dice throws, keeping track of the frequencies of oc¬ 
currence of each value when you compute the sum of two random integers between 1 
and 6. How large does /V have to be before your empirical results match the exact results 
to three decimal places? 


1.1.36 Empirical shuffle check. Run computational experiments to check that our 
shuffling code on page 32 works as advertised. Write a program Shuff 1 eTest that takes 
command-line arguments M and N, does N shuffles of an array of size M that is initial¬ 
ized with a[i ] = i before each shuffle, and prints an M-by-M table such that row i 
gives the number of times i wound up in position j for all j. All entries in the array 
should be close to N/M. 

1.1.37 Bad shuffling. Suppose that you choose a random integer between 0 and N-l 
in our shuffling code instead of one between i and N-l. Show that the resulting order is 
not equally likely to be one of the N! possibilities. Run the test of the previous exercise 
for this version. 

1.1.38 Binary search versus brute-force search. Write a program BruteForceSearch 
that uses the brute-force search method given on page 48 and compare its running time 
on your computer with that of Bi narySearch for 1 argeW.txt and 1 argeT.txt. 
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EXPERIMENTS (continued) 


1.1.39 Random matches. Write a BinarySearch client that takes an int value T as 
command-line argument and runs T trials of the following experiment for N= 10 3 , 10 4 , 
10 5 , and 10 6 : generate two arrays of N randomly generated positive six-digit i nt values, 
and find the number of values that appear in both arrays. Print a table giving the average 
value of this quantity over the T trials for each value of N. 
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1.2 DATA ABSTRACTION 


A data type is a set of values and a set of operations on those values. So far, we have 
discussed in detail Java’s primitive data types: for example, the values of the primitive 
data type i nt are integers between —2 31 and 2 31 — 1; the operations of i nt include +, *, 
-, /, %, <, and >. In principle, we could write all of our programs using only the built-in 
primitive types, but it is much more convenient to write programs at a higher level of 
abstraction. In this section, we focus on the process of defining and using data types, 
which is known as data abstraction (and supplements the function abstraction style that 
is the basis of section 1 . 1 ). 

Programming in Java is largely based on building data types known as reference types 
with the familiar Java class. This style of programming is known as object-oriented 
programming, as it revolves around the concept of an object, an entity that holds a data 
type value. With Java’s primitive types we are largely confined to programs that operate 
on numbers, but with reference types we can write programs that operate on strings, 
pictures, sounds, any of hundreds of other abstractions that are available in Java’s stan¬ 
dard libraries or on our booksite. Even more significant than libraries of predefined 
data types is that the range of data types available in Java programming is open-ended, 
because you can define your own data types to implement any abstraction whatsoever. 

An abstract data type (ADT) is a data type whose representation is hidden from the 
client. Implementing an ADT as a Java class is not very different from implementing a 
function library as a set of static methods. The primary difference is that we associate 
data with the function implementations and we hide the representation of the data 
from the client. When using an ADT, we focus on the operations specified in the API and 
pay no attention to the data representation; when implementing an ADT, we focus on 
the data, then implement operations on that data. 

Abstract data types are important because they support encapsulation in program 
design. In this book, we use them as a means to 

■ Precisely specify problems in the form of APIs for use by diverse clients 

■ Describe algorithms and data structures as API implementations 

Our primary reason for studying different algorithms for the same task is that perfor¬ 
mance characteristics differ. Abstract data types are an appropriate framework for the 
study of algorithms because they allow us to put knowledge of algorithm performance 
to immediate use: we can substitute one algorithm for another to improve performance 
for all clients without changing any client code. 
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Using abstract data types You do not need to know how a data type is imple¬ 
mented in order to be able to use it, so we begin by describing how to write programs 
that use a simple data type named Counter whose values are a name and a nonnega¬ 
tive integer and whose operations are create and initialize to zero, increment by one, and 
examine the current value. This abstraction is useful in many contexts. For example, it 
would be reasonable to use such a data type in electronic voting software, to ensure that 
the only thing that a voter can do is increment a chosen candidate’s tally by one. Or, 
we might use a Counter to keep track of fundamental operations when analyzing the 
performance of algorithms. To use a Counter, you need to learn our mechanism for 
specifying the operations defined in the data type and the Java language mechanisms 
for creating and manipulating data-type values. Such mechanisms are critically im¬ 
portant in modern programming, and we use them throughout this book, so this first 
example is worthy of careful attention. 

API for an abstract data type. To specify the behavior of an abstract data type, we use 
an application programming interface (API), which is a list of constructors and instance 
methods (operations), with an informal description of the effect of each, as in this API 
for Counter: 

public class Counter 

Counter(Stri ng id) create a counter named id 
i ncrementO increment the counter by one 

tal 1 y O number of increments since creation 

toStri ng() string representation 

An API fora counter 

Even though the basis of a data-type definition is a set of values, the role of the values 
is not visible from the API, only the operations on those values. Accordingly, an ADT 
definition has many similarities with a library of static methods (see page 24): 

■ Both are implemented as a Java class. 

■ Instance methods may take zero or more arguments of a specified type, sepa¬ 
rated by commas and enclosed in parentheses. 

■ They may provide a return value of a specified type or no return value (signified 
by voi d). 

And there are three significant differences: 

■ Some entries in the API have the same name as the class and lack a return type. 
Such entries are known as constructors and play a special role. In this case, 
Counter has a constructor that takes a Stri ng argument. 


voi d 
i nt 

Stri ng 
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■ Instance methods lack the stati c modifier. They are not static methods—their 
purpose is to operate on data type values. 

■ Some instance methods are present so as to adhere to Java conventions—we 
refer to such methods as inherited methods and shade them gray in the API. 

As with APIs for libraries of static methods, an API for an abstract data type is a con¬ 
tract with all clients and, therefore, the starting point both for developing any client 
code and for developing any data-type implementation. In this case, the API tells us 
that to use Counter, we have available the CounterO constructor, the incrementO 
and tally() instance methods, and the inherited toStri ng() method. 

Inherited methods. Various Java conventions enable a data type to take advantage of 
built-in language mechanisms by including specific methods in the API. For example, 
all Java data types inherit a toStri ng() method that returns a Stri ng representation 
of the data-type values. Java calls this method when any data-type value is to be concat¬ 
enated with a Stri ng value with the + operator. The default implementation is not par¬ 
ticularly useful (it gives a string representation of the memory address of the data-type 
value), so we often provide an implementation that overrides the default, and include 
toStri ng() in the API whenever we do so. Other examples of such methods include 
equal s(), compareToO, and hashCodeO (see page 101). 

Client code. As with modular programming based on static methods, the API allows 
us to write client code without knowing details of the implementation (and to write 
implementation code without knowing details of any particular client). The mecha¬ 
nisms introduced on page 28 for organizing programs as independent modules are use¬ 
ful for all Java classes, and thus are effective for modular programming with ADTs as 
well as for libraries of static methods. Accordingly, we can use an ADT in any program 
provided that the source code is in a . j ava file in the same directory, or in the standard 
Java library, or accessible through an i mport statement, or through one of the classpath 
mechanisms described on the booksite. All of the benefits of modular programming 
follow. By encapsulating all the code that implements a data type within a single Java 
class, we enable the development of client code at a higher level of abstraction. To de¬ 
velop client code, you need to be able to declare variables, create objects to hold data¬ 
type values, and provide access to the values for instance methods to operate on them. 
These processes are different from the corresponding processes for primitive types, 
though you will notice many similarities. 
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Objects. Naturally, you can declare that a variable heads is to be associated with data 
of type Counter with the code 
Counter heads; 


one Counter object 


heads 


460 


460 


two Counter objects 


but how can you assign values or specify operations? The answer to this question in¬ 
volves a fundamental concept in data abstraction: an object is an entity that can take on 
a data-type value. Objects are characterized by three essential prop¬ 
erties: state, identity, and behavior. The state of an object is a value 
from its data type. The identity of an object distinguishes one object 
from another. It is useful to think of an object’s identity as the place 
where its value is stored in memory. The behavior of an object is the 
effect of data-type operations. The implementation has the sole re¬ 
sponsibility for maintaining an object’s identity, so that client code 
can use a data type without regard to the representation of its state 
by conforming to an API that describes an object’s behavior. An ob¬ 
ject’s state might be used to provide information to a client or cause 
a side effect or be changed by one of its data type’s operations, but 
the details of the representation of the data-type value are not rel¬ 
evant to client code. A reference is a mechanism for accessing an ob¬ 
ject. Java nomenclature makes clear the distinction from primitive 
types (where variables are associated with values) by using the term 
reference types for nonprimitive types. The details of implementing 
references vary in Java implementations, but it is useful to think of a 
reference as a memory address, as shown at right (for brevity, we use 
three-digit memory addresses in the diagram). 

Creating objects. Each data-type value is stored in an object. To 
create (or instantiate) an individual object, we invoke a constructor 
by using the keyword new, followed by the class name, followed by 
C) (or a list of argument values enclosed in parentheses, if the con¬ 
structor takes arguments). A constructor has no return type because 
it always returns a reference to an object of its data type. Each time 
that a client uses new(), the system 

■ Allocates memory space for the object 

■ Invokes the constructor to initialize its value 

■ Returns a reference to the object 

In client code we typically create objects in an initializing declaration that associates a 
variable with the object, as we often do with variables of primitive types. Unlike primi¬ 
tive types, variables are associated with references to objects, not the data-type values 


„ reference 


identity 
(details hidden 


s' 


heads 
tai 1 s 


460 


612 



Object representation 
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themselves. We can create any num¬ 
ber of objects from the same class— 
each object has its own identity 
and may or may not store the same 
value as another object of the same 
type. For example, the code 


declaration to associate call on constructor 

variable with object reference to create an object 


Counter heads 


new Counter("heads"); 


Creating an object 


Counter heads = new Counter("heads"); 
Counter tails = new Counter("tai1s"); 


creates two different Counte r objects. In an abstract data type, details of the representa¬ 
tion of the value are hidden from client code. You might assume that the value associ¬ 
ated with each Counter object is a Stri ng name and an i nt tally, but you cannot write 
code that depends on any specific representation (or even know whether that assumption 
is true—perhaps the tally is a 1 ong value). 


Counter heads; 

with new (constructor) 

heads 


- declaration 


object name 


as an expression 


Invoking instance methods. The purpose of an instance method is to operate on data¬ 
type values, so the Java language includes a special mechanism to invoke instance meth¬ 
ods that emphasizes a connection to an object. Specifically, we invoke an instance meth¬ 
od by writing a variable name that refers to an object, 
followed by a period, followed by an instance method 
name, followed by 0 or more arguments, enclosed in 
parentheses and separated by commas. An instance 
method might change the data-type value or just exam¬ 
ine the data-type value. Instance methods have all of 
the properties of static methods that we considered on 
page 24—arguments are passed by value, method names 
can be overloaded, they may have a return value, and 
they may cause side effects—but they have an addi¬ 
tional property that characterizes them: each invoca¬ 
tion is associated with an object. For example, the code 


new Counter ("heads") 


invoke a constructor (create an object ) 

as a statement (void return value) 


heads|. i ncrementO; 

T 


\ 


invoke an instance method 
that changes the object’s value 


heads|. tallyQ 


tai 1 s. tally() 


object name 


\ 

invoke an instance method 
that accesses the object’s value 


via automatic type conversion (toStringO) 

StdOut. pri ntl n( |heads| ) ; 

1 

invoke heads. toStringO 


Invoking instance methods 


heads. i ncrementO ; 

invokes the instance method i ncrementO to operate 
on the Counter object heads (in this case the opera¬ 
tion involves incrementing the tally), and the code 

heads.tallyO - tails.tallyO ; 

invokes the instance method tallyO twice, first to 
operate on the Counter object heads and then to op¬ 
erate on the Counter object tails (in this case the 
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operation involves returning the tally as an i nt value). As these examples illustrate, you 
can use calls on instance methods in client code in the same way as you use calls on stat¬ 
ic methods—as statements (voi d methods) or values in expressions (methods that re¬ 
turn a value). The primary purpose of stat¬ 
ic methods is to implement functions; the 
primary purpose of non-static (instance) 
methods is to implement data-type opera¬ 
tions. Either type of method may appear in 
client code, but you can easily distinguish 
between them, because a static method 
call starts with a class name (uppercase, by 
convention) and a non-static method call 
always starts with an object name (lower¬ 
case, by convention). These differences are 
summarized in the table at right. 



instance method 

static method 

sample call 

head. i ncrementO 

Math . sqrt(2.0) 

invoked with 

object name 

class name 


reference to object 


parameters 

and argument(s) 

argument (s) 

primary 

examine or change 

compute return 

purpose 

object value 

value 


Instance methods versus static methods 


Using objects. Declarations give us variable names for objects that we can use in code 
not just to create objects and invoke instance methods, but also in the same way as we 
use variable names for integers, floating-point numbers, and other primitive types. To 
develop client code for a given data type, w: 

■ Declare variables of the type, for use in referring to objects 

■ Use the keyword new to invoke a constructor that creates objects of the type 

■ Use the object name to invoke instance methods, either as statements or within 
expressions 

For example, the class FI i ps shown at the top of the next page is a Counter client that 
takes a command-line argument T and simulates T coin flips (it is also a StdRandom cli¬ 
ent). Beyond these direct uses, we can use variables associated with objects in the same 
way as we use variables associated with primitive-type values: 

■ In assignment statements 

■ To pass or return objects from methods 

■ To create and use arrays of object. 

Understanding the behavior of each of these types of uses requires thinking in terms of 
references, not values, as you will see when we consider them, in turn. 


Assignment statements. An assignment statement with a reference type creates a copy 
of the reference. The assignment statement does not create a new object, just another 
reference to an existing object. This situation is known as aliasing: both variables refer 
to the same object. The effect of aliasing is a bit unexpected, because it is different for 
variables holding values of a primitive type. Be sure that you understand the difference. 
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public class Flips 

{ 

public static void main(String[] args) 

{ 

int T = Integer.parselnt(args[0]); 
Counter heads = new Counter("heads"); 
Counter tails = new Counter("taiIs"); 
for (int t = 0; t < T; t++) 

if (StdRandom.bernoulli(0.5)) 
heads. i ncrementO ; 
else tai 1 s.incrementO ; 
StdOut.println(heads); 
StdOut.println(tails); 
int d = heads, tai ly() - tai Is. tall y() I 
StdOut.println("delta: " + Math.abs(d)); 

} 

} 

Counter client that simulates T coin flips 


% java Flips 10 
5 heads 
5 tails 
delta: 0 

% java Flips 10 
8 heads 
2 tails 
delta: 6 

% java Flips 1000000 
499710 heads 
500290 tails 
delta: 580 


If x and y are variables of a primitive type, then the as¬ 
signment x = y copies the value of y to x. For reference 
types, the reference is copied (not the value). Aliasing is a 
common source of bugs in Java programs, as illustrated 
by the following example: 

Counter cl = new Counter("ones"); 
cl. i ncrementO ; 

Counter c2 = cl; 
c2. i ncrementO ; 

StdOut.println(cl); 

With a typical toStringO implementation this code 
would print the string "2 ones" which may or may not 
be what was intended and is counterintuitive at first. Such 
bugs are common in programs written by people without 
much experience in using objects (that maybe you, so pay 
attention here!). Changing the state of an object impacts 
all code involving aliased variables referencing that ob¬ 
ject. We are used to thinking of two different variables of 
primitive types as being independent, but that intuition 
does not carry over to variables of reference types. 


Counter cl; 

cl = new Counter("ones"); 
cl. i ncrementO; 

Counter c2 = cl; 
c2 . i ncrementO ; 



references to 
same object 


reference to 
"ones" 
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Objects as arguments. You can pass objects as arguments to methods. This ability typi¬ 
cally simplifies client code. For example, when we use a Counter as an argument, we are 
essentially passing both a name and a tally, but need only specify one variable. When 
we call a method with arguments, the effect in Java is as if each argument value were 
to appear on the right-hand side of an assignment statement with the corresponding 
argument name on the left. That is, Java passes a copy of the argument value from the 
calling program to the method. This arrangement is known as pass by value (see page 
24). One important consequence is that the method cannot change the value of a caller’s 
variable. For primitive types, this policy is what we expect (the two variables are inde¬ 
pendent), but each time that we use a reference type as a method argument we create 
an alias, so we must be cautious. In other words, the convention is to pass the reference 
by value (make a copy of it) but to pass the object by reference. For example, if we pass 
a reference to an object of type Counter, the method cannot change the original refer¬ 
ence (make it point to a different Counter), but it can change the value of the object, 
for example by using the reference to call i ncrement(). 

Objects as return values. Naturally, you can also use an object as a return value from 
a method. The method might return an object passed to it as an argument, as in the 
example below, or it might create an object and return a reference to it. This capa¬ 
bility is important because 
Java methods allow only one 
return value—using objects 
enables us to write code that, 
in effect, returns multiple 
values. 


Example of a static method with object arguments and return values 


% java FlipsMax 1000000 
500281 tails wins 


public class FlipsMax 
{ 

public static Counter max(Counter x, Counter y) 

{ 

if (x.tallyO > y.tallyO) return x; 
else return y; 

} 

public static void main(String[] args) 

{ 

int T = Integer.parselnt(args[0]); 

Counter heads = new Counter("heads"); 

Counter tails = new Counter("taiIs"); 
for (int t = 0; t < T; t++) 

if (StdRandom.bernoul1i(0.5)) 
heads.increment(); 
else tails.increment() ; 

if (heads.tallyO == tails.tallyO) 
Std0ut.println("Tie") ; 

else StdOut.println(max(heads, tails) + " wins"); 

} 

} 
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Arrays are objects. In Java, every value of any nonprimitive type is an object. In par¬ 
ticular, arrays are objects. As with strings, there is special language support for certain 
operations on arrays: declarations, initialization, and indexing. As with any other ob¬ 
ject, when we pass an array to a method or use an array variable on the right hand side 
of an assignment statement, we are making a copy of the array reference, not a copy 
of the array. This convention is appropriate for the typical case where we expect the 
method to be able to modify the array, by rearranging its entries, as, for example, in 
java, uti 1 .Arrays. sort() or the shuffl e() method that we considered on page 32. 

Arrays of objects. Array entries can be of any type, as we have already seen: args [] in 
our mai n () implementations is an array of St r i ng objects. When we create an array of 
objects, we do so in two steps: 

■ Create the array, using the bracket syntax for array constructors. 

■ Create each object in the array, using a standard constructor for each. 

For example, the code below simulates rolling a die, using an array of Counter objects 
to keep track of the number of occurrences of each possible value. An array of objects 
in Java is an array of references to objects, not the objects themselves. If the objects are 
large, then we may gain efficiency by not having to move them around, just their refer¬ 
ences. If they are small, we may lose efficiency by having to follow a reference each time 
we need to get to some information. 


public class Rolls 
{ 

public static void main(String[] args) 

{ 

int T = Integer.parselnt(args[0]); 
int SIDES = 6; 

Counter[] rolls = new Counter[SIDES+l]; 
for (int i = 1; i <= SIDES; i++) 
rolls[i] = new Counter(i + '"s"); 


for (int t = 0; t < T; t++) 

{ 

int result = StdRandom.uniform(l, SIDES+1); 
rolls[result],increment() ; 

} 

for (int i = 1; i <= SIDES; i++) 
StdOut.println(rolls[i]) ; 

} 

} 


Counter client that simulates T rolls of a die 


% java Rolls 1000000 

167308 l's 

166540 2 ' s 

166087 3' s 

167051 4' s 

166422 5 ' s 

166592 6's 
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with this focus on objects, writing code that embraces data abstraction (defining 
and using data types, with data-type values held in objects) is widely referred to as 
object-oriented programming. The basic concepts that we have just covered are the start¬ 
ing point for object-oriented programming, so it is worthwhile to briefly summarize 
them. A data type is a set of values and a set of operations defined on those values. We 
implement data types in independent Java class modules and write client programs 
that use them. An object is an entity that can take on a data-type value or an instance of 
a data type. Objects are characterized by three essential properties: state, identity, and 
behavior. A data-type implementation supports clients of the data type as follows: 

■ Client code can create objects (establish identity) by using the new construct to 
invoke a constructor that creates an object, initializes its instance variables, and 
returns a reference to that object. 

■ Client code can manipulate data-type values (control an object’s behavior, pos¬ 
sibly changing its state) by using a variable associated with an object to invoke 
an instance method that operates on that object’s instance variables. 

■ Client code can manipulate objects by creating arrays of objects and passing them 
and returning them to methods, in the same way as for primitive-type values, 
except that variables refer to references to values, not the values themselves. 

These capabilities are the foundation of a flexible, modern, and widely useful program¬ 
ming style that we will use as the basis for studying algorithms in this book. 
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Examples of abstract data types The Java language has thousands of built-in 
ADTs, and we have defined many other ADTs to facilitate the study of algorithms. In¬ 
deed, every Java program that we write is a data-type implementation (or a library of 
static methods). To control complexity, we will specifically cite APIs for any ADT that 
we use in this book (not many, actually). 

In this section, we introduce as examples several data types, with some examples 
of client code. In some cases, we present excerpts of APIs that may contain dozens of 
instance methods or more. We articulate these APIs to present real-world examples, to 
specify the instance methods that we will use in the book, and to emphasize that you 
do not need to know the details of an ADT implementation in order to be able to use it. 

For reference, the data types that we use and develop in this book are shown on the 
facing page. These fall into several different categories: 

■ Standard system ADTs in j ava. 1 ang. *, which can be used in any Java program. 

■ Java ADTs in libraries such as j ava. awt, j ava. net, and j ava. i o, which can also 
be used in any Java program, but need an i mport statement. 

■ Our I/O ADTs that allow us to work with multiple input/output streams similar 
to Stdln and StdOut. 

■ Data-oriented ADTs whose primary purpose is to facilitate organizing and pro¬ 
cessing data by encapsulating the representation. We describe several examples 
for applications in computational geometry and information processing later in 
this section and use them as examples in client code later on. 

■ Collection ADTs whose primary purpose is to facilitate manipulation collections 
of data of the same. We describe the basic Bag, Stack, and Queue types in sec¬ 
tion 1 . 3 , PQ types in chapter 2 , and the ST and SET types in chapters 3 and 5. 

■ Opertions-oriented ADTs that we use to analyze algorithms, as described in sec¬ 
tion 1.4 and section 1 . 5 . 

■ ADTs for graph algorithms, including both data-oriented ADTs that focus on 
encapsulating representations of various kinds of graphs and operations-orient- 
ed ADTs that focus on providing specifications for graph-processing algorithms. 

This list does not include the dozens of types that we consider in exercises, which may 
be found in the index. Also, as described on page 90, we often distinguish multiple imple¬ 
mentations of various ADTs with a descriptive prefix. As a group, the ADTs that we 
use demonstrate that organizing and understanding the data types that you use is an 
important factor in modern programming. 

A typical application might use only five to ten of these ADTs. A prime goal in the 
development and organization of the ADTs in this book is to enable programmers to 
easily take advantage of a relatively small set of them in developing client code. 
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standard Java system types in java.lang 

collection types 


Integer 

i nt wrapper 

Stack 

pushdown stack 

Double 

doubl e wrapper 

Queue 

FIFO queue 

Stri ng 

indexed chars 

Bag 

bag 

Stri ngBui 1 der 

builder for strings 

MinPQ MaxPQ 

priority queue 

other Java types 


IndexMinPQ IndexMinPQ 

priority queue ( indexed) 

java.awt.Color 

colors 

ST 

symbol table 

java.awt.Font 

fonts 

SET 

set 

java.net.URL 

URLs 

Stri ngST 

symbol table (string keys) 

java. io. Fi 1 e 

files 

data-oriented graph types 


our standard I/O types 


Graph 

graph 

In 

input stream 

Digraph 

directed graph 

Out 

output stream 

Edge 

edge ( weighted ) 

Draw 

drawing 

EdgeWeightedGraph 

graph ( weighted ) 

data-oriented types for client examples 

Di rectedEdge 

edge ( directed, weighted) 

Point2D 

point in the plane 

EdgeWeightedDigraph 

graph ( directed, weighted) 

Interval ID 

ID interval 

operations-oriented graph types 


Interval 2D 

2D interval 

UF 

dynamic connectivity 

Date 

date 

DepthFirstPaths 

DFS path searcher 

Transaction 

transaction 

CC 

connected components 

types for the analysis of algorithms 

BreadthFirstPaths 

BFS path search 

Counter 

counter 

DirectedDFS 

DFS digraph path search 

Accumulator 

accumulator 

DirectedBFS 

BFS digraph path search 

VisualAccumulator 

visual version 

TransitiveClosure 

all paths 

Stopwatch 

stopwatch 

Topological 

topological order 



DepthFirstOrder 

DFS order 



DirectedCycle 

cycle search 



see 

strong components 



MST 

minimum spanning tree 



SP 

shortest paths 


Selected ADTs used in this book 
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Geometric objects. A natural example of object-oriented programming is design¬ 
ing data types for geometric objects. For example, the APIs on the facing page define 

abstract data types for three familiar 


public 
{ 


static void main(String[] args) 


double xlo = Double.parseDouble(args[0]); 
double xhi = Double.parseDouble(args[1]); 
double ylo = Double. parseDouble(args[2]) ; 
double yhi = Double.parseDouble(args[3]); 
int T = Integer.parselnt(args[4]); 


Interval ID x = new IntervallD(xlo, 
Interval ID y = new IntervallD(ylo, 
Interval2D box = new Interval2D(x, 
box.drawQ ; 


xhi); 

yhi); 

y); 


Counter c = new Counter(“hits”); 
for (int t = 0; t < T; t++) 

{ 

double x = Math.random(); 
double y = Math.random(); 

Point p = new Point(x, y) ; 
if (box.contains(p)) c.incrementO ; 
else p.draw(); 

} 

StdOut.println(c) ; 
StdOut.println(box.areaO) ; 


Interval2D test client 



% java Interval 2D .2 .5 

297 hits 

.03 


5 .6 10000 


geometric objects: Point2D (points 
in the plane), Interval ID (intervals 
on the line), and Interval 2D (two- 
dimensional intervals in the plane, or 
axis-aligned rectangles). As usual, the 
APIs are essentially self-documenting 
and lead immediately to easily under¬ 
stood client code such as the example 
at left, which reads the boundaries 
of an Interval 2D and an integer T 
from the command line, generates T 
random points in the unit square, and 
counts the number of points that fall 
in the interval (an estimate of the area 
of the rectangle). For dramatic effect, 
the client also draws the interval and 
the points that fall outside the inter¬ 
val. This computation is a model for 
a method that reduces the problem 
of computing the area and volume 
of geometric shapes to the problem 
of determining whether a point falls 
within the shape or not (a less difficult but not trivial prob¬ 
lem). Of course, we can define APIs for other geometric ob¬ 
jects such as line segments, triangles, polygons, circles, and 
so forth, though implementing operations on them can be 
challenging. Several examples are addressed in the exercises 
at the end of this section. 

PROGRAMS THAT PROCESS GEOMETRIC OBJECTS have wide 
application in computing with models of the natural world, 
in scientific computing, video games, movies, and many 
other applications. The development and study of such pro¬ 
grams and applications has blossomed into a far-reaching 
field of study known as computational geometry, which is a 
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public class Point2D 


Point2D(double x, double y) 

create a point 

double x() 

x coordinate 

double y() 

y coordinate 

double r() 

radius (polar coordinates) 

double theta() 

angle (polar coordinates) 

double di stTo(Point2D that) 

Euclidean distance from this point to that 

void drawQ 

draw the point on StdDraw 

An API for points in the plane 

public class IntervallD 


IntervallD(double lo, double hi) 

create an interval 

double lengthC) 

length of the interval 

boolean contains(double x) 

does the interval contain x? 

boolean intersects(IntervallD that) 

does the interval intersect that? 

void drawQ 

draw the interval on StdDraw 


An API for intervals on the line 


public class Interval2D 



Interval2D(IntervalID x, IntervallD y) 

create a 2D interval 

double 

area() 

area of the 2D interval 

boolean 

contains(Point p) 

does the 2D interval contain p? 

boolean 

intersects(Interval 2D that) 

does the 2D interval intersect that? 

voi d 

drawQ 

draw the 2D interval on StdDraw 


An API for two dimensional intervals in the plane 


fertile area of examples for the application of the algorithms that we address in this 
book, as you will see in examples throughout the book. In the present context, our 
interest is to suggest that abstract data types that directly represent geometric abstrac¬ 
tions are not difficult to define and can lead to simple and clear client code. This idea is 
reinforced in several exercises at the end of this section and on the booksite. 
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Information processing Whether it be a bank processing millions of credit card trans¬ 
actions or a web analytics company processing billions of touchpad taps or a scien¬ 
tific research group processing millions of experimental observations, a great many 
applications are centered around processing and organizing information. Abstract data 
types provide a natural mechanism for organizing the information. Without getting 
into details, the two APIs on the facing page suggest a typical approach for a commer¬ 
cial application. The idea is to define data types that allow us to keep information in 
objects that correspond to things in the real world. A date is a day, a month, and a year 
and a transaction is a customer, a date, and an amount. These two are just examples: we 
might also define data types that can hold detailed information for customers, times, 
locations, goods and services, or whatever. Each data type consists of constructors that 
create objects containing the data and methods for use by client code to access it. To 
simplify client code, we provide two constructors for each type, one that presents the 
data in its appropriate type and another that parses a string to get the data (see exer¬ 
cise 1.2.19 fd r details). As usual, there is no reason for client code to know the rep¬ 
resentation of the data. Most often, the reason to organize the data in this way is to 
treat the data associated with an object as a single entity: we can maintain arrays of 
Transacti on values, use Date values as a argument or a return value for a method, and 
so forth. The focus of such data types is on encapsulating the data, while at the same 
time enabling the development of client code that does not depend on the representa¬ 
tion of the data. We do not dwell on organizing information in this way, except to take 
note that doing so and including the inherited methods toStringO, compareToO, 
equal s(), and hashCodeO allows us to take advantage of algorithm implementations 
that can process any type of data. We will discuss inherited methods in more detail 
on page 100. For example, we have already noted Java’s convention that enables clients 
to print a string representation of every value if we include toStringO implemen¬ 
tation in a data type. We consider conventions corresponding to the other inherited 
methods in section 1.3, section 2.5, section 3.4, and section 3.5, using Date and 
Transacti on as examples, section 1.3 gives classic examples of data types and a Java 
language mechanism known as parameterized types, or generics, that takes advantage of 
these conventions, and chapter 2 and chapter 3 are also devoted to taking advantage 
of generic types and inherited methods to develop implementations of sorting and 
searching algorithms that are effective for any type of data. 

whenever you have data of different types that logically belong together, it is 
worthwhile to contemplate defining an ADT as in these examples. The ability to do so 
helps to organize the data, can greatly simplify client code in typical applications, and 
is an important step on the road to data abstraction. 
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public class Date implements Comparable<Date> 

Date(int month, int day, int year) 
Date(String date) 
int month() 
int day() 
int year() 

String toStringO 
boolean equals(Object that) 
int compareTo(Date that) 
int hashCodeQ 


create a date 

create a date (parse constructor) 

month 

day 

year 

string representation 
is this the same date as that? 
compare this date to that 
hash code 


public class Transaction implements Comparable<Transaction> 


String 
Date 
double 
String 
boolean 
i nt 
i nt 


Transaction(String who, Date when, double amount) 


Transaction(String transaction) 

who() 

when() 

amountO 

toStri ng() 

equals(Object that) 

compareTo(Transaction that) 

hashCodeQ 


create a transaction (parse constructor) 

customer name 

date 

amount 

string representation 
is this the same transaction as that? 
compare this transaction to that 
hash code 


Sample APIs for commercial applications (dates and transactions) 
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Strings. Java’s String is an important and useful ADT. A String is an indexed se¬ 
quence of char values. Stri ng has dozens of instance methods, including the following: 


public class String 



Stri ng() 

create an empty string 

i nt 

1ength() 

length of the string 

i nt 

charAt(int i) 

i th character 

i nt 

indexOf(String p) 

first occurrence of p (-1 if none) 

i nt 

indexOf(String p, int i) 

first occurrence of p after i (-1 if none) 

Stri ng 

concat(String t) 

this string with t appended 

Stri ng 

substri ng(int i, int j) 

substring of this string (i th to j -1st chars) 

Stri ng [] 

split(String delim) 

strings between occurrences o/del i m 

i nt 

compareTo(String t) 

string comparison 

boolean 

equals(String t) 

is this string’s value the same as ts ? 

i nt 

hashCodeO 

hash code 


Java String API (partial list of methods) 


String values are similar to arrays of characters, but the two are not the same. Ar¬ 
rays have built-in Java language syntax for accessing a character; String has instance 
methods for indexed access, length, and many other operations. On the other hand, 
String has special language support for initialization and concatenation: instead of 
creating and initializing a string with a constructor, we can use a string literal; instead 
of invoking the method concatO we can use the + operator. We do not need to con¬ 
sider the details of the implementation, though 
understanding performance characteristics of 
some of the methods is important when develop¬ 
ing string-processing algorithms, as you will see 
in chapter 5 . Why not just use arrays of charac¬ 
ters instead of Stri ng values? The answer to this 
question is the same as for any ADT: to simplify 
and clarify client code. With Stri ng, we can write 
clear and simple client code that uses numerous 
convenient instance methods without regard to 
the way in which strings are represented (see fac¬ 
ing page). Even this short list contains powerful 

operations that require advanced algorithms such Examples of string operations 


String a 
String b 
String c 


now is ; 
"the time 
"to" 


call 

value 

a.lengthO 

7 

a.charAt(4) 

i 

a.concat(c) 

"now i 

a.indexOf("is") 

4 

substring(2, 5) 

"w i" 

a. spl i t(" ") [0] 

"now" 

a.split(" ")[1] 

"i s" 

b.equals(c) 

fal se 
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task 

implementation 

is the string 
a palindrome? 

public static boolean isPali ndrome(String s) 

{ 

i nt N = s. 1 ength() ; 

for (int i =0; i < N/2; i++) 

if (s.charAt(i) != s.charAt(N-l-i)) 
return false; 
return true; 

} 

extract file name 
and extension from a 
command-line 
argument 

Stri ng s = args [0]; 
int dot = s.rank("."); 

String base = s.substring(0, dot); 

String extension = s. substri ng(dot + 1, s.lengthO); 

print all lines in 
standard input that 
contain a string 
specified on the 
command line 

String query = args[0]; 
while (! Stdln.isEmptyO) 

{ 

String s = Stdln . readLi ne() ; 

if (s .contai ns(query)) StdOut.println(s) ; 

} 


create an array 
of the strings on Stdln 
delimited by whitespace 

String input = Stdln . readAl 1 () ; 

Stringf] words = input.split("\\s+") ; 

check whether an array 
of strings is in 
alphabetical order 

public boolean isSorted(Stri ng[] a) 

{ 

for (int i = 1; i < a.length; i++) 

{ 

if (a[i-l] .compareTo(a[i]) > 0) 
return false; 

} 

return true; 

} 

Typical string-processing code 
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as those considered in chapter 5 . For example, the argument of splitC) can be a 
regular expression (see section 5 . 4 )—the splitO example on page 81 uses the argu¬ 
ment "\\s+", which means “one or more tabs, spaces, newlines, or returns.” 

Input and output revisited. A disadvantage of the Stdln, StdOut, and StdDraw stan¬ 
dard libraries of section 1.1 is that they restrict us to working with just one input file, 
one output file, and one drawing for any given program. With object-oriented pro¬ 
gramming, we can define similar mechanisms that allow us to work with multiple input 
streams, output streams, and drawings within one program. Specifically, our standard 
libary includes the data types In, Out, and Draw with the APIs shown on the facing page, 
When invoked with a constructor having a Stri ng argument, In and Out will first try 
to find a file in the current directory of your computer that has that name. If it cannot 

do so, it will assume the argu¬ 
ment to be a website name and 
will try to connect to that web¬ 
site (if no such website exists, it 
will issue a runtime exception). 
In either case, the specified file 
or website becomes the source/ 
target of the input/output for 
the stream object thus created, 
and the read*() and print*() 
methods will refer to that file or 
website. (If you use the no-argu¬ 
ment constructor, then you ob¬ 
tain the standard streams.) This 
arrangement makes it possible 
for a single program to process 
multiple files and drawings. You also can assign such 
objects to variables, pass them as arguments or re¬ 
turn values from methods, create arrays of them, and 
manipulate them just as you manipulate objects of 
any type. The program Cat shown at left is a sample 
client of In and Out that uses multiple input streams 
to concatenate several input files into a single out¬ 
put file. The In and Out classes also contain static 
methods for reading files containing values that are 
all int, double, or String types into an array (see 
page 126 and exercise 1 . 2 . 15 ). 


% more inl.txt 
Thi sis 

% more in2.txt 
a ti ny 
test. 

% java Cat inl.txt in2.txt out.txt 

% more out.txt 
This is 
a ti ny 
test. 


public class Cat 
{ 

public static void main (Stri ng[] args) 

{ // Copy input files to out (last argument). 

Out out = new Out(args[args.length-1]); 
for (int i =0; i < args.length - 1; i++) 

{ // Copy input file named on ith arg to out. 

In in = new In(args[i]); 

String s = in. readAl 1 () ; 
out.println(s); 
i n . close() ; 

} 

out.close(); 

} 

} 

A sample In and Out client 
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public class In 



In() 

create an input stream from standard input 


In(String name) 

create an input stream from a file or website 

boolean 

i sEmptyO 

true if no more input, fal se otherwise 

i nt 

readlntO 

read a value of type i nt 

double 

readDouble() 

read a value of type doubl e 

voi d 

cl ose() 

close the input stream 

Note: all operations supported by Stdln are 

also supported for In objects. 


API for our data type for input streams 

public class 

Out 




OutO 

create an output stream to standard output 


Out(String name) 

create an output stream to a file 

voi d 

print(String s) 

append s to the output stream 

voi d 

println(String s) 

append s and a newline to the output stream 

voi d 

pri ntl n() 

append a newline to the output stream 

voi d 

printf (String f, . . .) 

formatted print to the output steam 

voi d 

cl ose() 

close the output stream 


Note: all operations supported by StdOut are also supported for Out objects. 

API for our data type for output streams 


public class Draw 
Draw() 

void line(double xO, double yO, double xl, double yl) 
void point(double x, double y) 

Note: all operations supported by StdDraw are also supported for Draw objects. 

API for our data type for drawings 
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Implementing an abstract data type. As with libraries of static methods, we 
implement ADTs with a Java cl ass, putting the code in a file with the same name as 
the class, followed by the .java extension. The first statements in the file declare in¬ 
stance variables that define the data-type values. Following the instance variables are the 
constructor and the instance methods that implement operations on data-type values. 
Instance methods may be public (specified in the API) or private (used to organize the 
computation and not available to clients). A data-type definition may have multiple 
constructors and may also include definitions of static methods. In particular, a unit- 
test client mai n() is normally useful for testing and debugging. As a first example, we 
consider an implementation of the Counter ADT that we defined on page 65. A full 
annotated implementation is shown on the facing page, for reference as we discuss its 
constituent parts. Every ADT implementation that you will develop has the same basic 
ingredients as this simple example. 


instance 

variable 

declarations 


public class Counter 
i 

private final String name; 
private int count; 


} 

Instance variables in ADTs are private 


Instance variables. To define data-type 
values (the state of each object), we de¬ 
clare instance variables in much the same 
way as we declare local variables. There is a 
critical distinction between instance vari¬ 
ables and the local variables within a static 

method or a block that you are accustomed to: there is just one value corresponding to 
each local variable at a given time, but there are numerous values corresponding to each 
instance variable (one for each object that is an instance of the data type). There is no 
ambiguity with this arrangement, because each time that we access an instance variable, 
we do so with an object name—that object is the one whose value we are accessing. 
Also, each declaration is qualified by a visibility modifier. In ADT implementations, we 
use pri vate, using a Java language mechansim to enforce the idea that the representa¬ 
tion of an ADT is to be hidden from the client, and also f i nal , if the value is not to be 
changed once it is initialized. Counter has two instance variables: a Stri ng value name 
and an i nt value count. If we were to use publ i c instance variables (allowed in Java) 
the data type would, by definition, not be abstract, so we do not do so. 


Constructors. Every Java class has at least one constructor that establishes an object’s 
identity. A constructor is like a static method, but it can refer directly to instance vari¬ 
ables and has no return value. Generally, the purpose of a constructor is to initialize 
the instance variables. Every constructor creates an object and provides to the client a 
reference to that object. Constructors always share the same name as the class. We can 
overload the name and have multiple constructors with different signatures, just as 
with methods. If no other constructor is defined, a default no-argument constructor is 


www.it-ebooks.info 



1.2 ■ 


Data Abstraction 


85 


instance 

variables 


constructor 


public class Counter 

{ 

private final String name;| 
private int count; 



class 

name 


public Counter (String id) 
{ name = id; } 


instance 

methods 



public void increment() 
{ count++; } 


public int tallyO 
{ return count; } 


public String toStringO 
{ return count + " " + name; } 


instance 

variable 

name 


test client - 


create 
and c 
initialize 
objects 


public static void main(String[] args) 

{ 

— Counter heads = new Counter("heads"); 
— Counter tails = new |Counter("tai 1 s") ;| 


heads. i ncrementO ; 
heads. i ncrementO I 
tai 1 s.incrementO I 


\ invoke 
constructor 


automatically invoke 
z toStringO 


object 


StdOut. printl n (heads + " " + t ai 1 s) ; /' r 
StdOut.println(heads.tallyO + |tails.tally() | ) 

\ invoke 


method 


Anatomy of a class that defines a data type 
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implicit, has no arguments, and initializes instance values to default values. The default 
values of instance variables are 0 for primitive numeric types, fal se for bool ean, and 
null for reference types. These defaults 
may be changed by using initializing 
declarations for instance variables. Java 
automatically invokes a constructor 
when a client program uses the keyword 
new. Overloaded constructors are typi¬ 
cally used to initialize instance variables 
to client-supplied values other than the 
defaults. For example, Counter has a 
one-argument constructor that initial¬ 
izes the name instance variable to the 
value given as argument (leaving the 
count instance variable to be initialized 
to the default value 0). 

Instance methods. To implement data-type instance methods (the behavior of each 
object), we implement instance methods with code that is precisely like the code that you 
learned in section i.i to implement static methods (functions). Each instance method 
has a return type, a signature (which specifies its name and the types and names of its 
parameter variables), and a body (which 
consists of a sequence of statements, in¬ 
cluding a return statement that provides 
a value of the return type back to the cli¬ 
ent). When a client invokes a method, the 
parameter values (if any) are initialized 
with client values, the statements are ex¬ 
ecuted until a return value is computed, 
and the value is returned to the client, 
with the same effect as if the method in¬ 
vocation in the client were replaced with that value. All of this action is the same as for 
static methods, but there is one critical distinction for instance methods: they can access 
and perform operations on instance variables. How do we specify which object’s instance 
variables we want to use? If you think about this question for a moment, you will see 
the logical answer: a reference to a variable in an instance method refers to the value 
for the object that was used to invoke the method. When we say heads. i ncremen t() the 
code in i ncrementQ is referring to the instance variables for heads. In other words, 


visibility return 
modifier type 

l i 


method 

name 


^ signature 


publ ic||void ipTncrementO] 


{ |count[f+; } 

\ 

instance variable name 

Anatomy of an instance method 


public class Counter 
{ 

private final String name; 
private int count; 


visibility pjQ re turn constructor name 
modifier tvDe (same as class name) 


\ 


type 


parameter 
- variable 


publ i c | |Counter] ( | Stri ng i d | ) 


{|name = id; | } 

\ 

code to initialize instance variables 
(count initialized to 0 by default ) 


signature 


Anatomy of a constructor 
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object-oriented programming adds one critically important additional way to use vari¬ 
ables in a Java program: 

■ to invoke an instance method that operates on the object’s values. 

The difference from working solely with static methods is semantic (see the Q&A), 
but has reoriented the way that modern programmers think about developing code in 
many situations. As you will see, it also dovetails well with the study of algorithms and 
data structures. 

Scope. In summary, the Java code that we write to implement instance methods uses 
three kinds of variables: 

■ Parameter variables 

■ Local variables 

■ Instance variables 

The first two of these are the same as for static methods: parameter variables are spec¬ 
ified in the method signature and initialized with client values when the method is 
called, and local variables are declared and initialized within the method body. The 
scope of parameter variables is the entire method; the scope of local variables is the 
following statements in the block where they are defined. Instance variables are com¬ 
pletely different: they hold data-type values for objects in a class, and their scope is the 
entire class (whenever there is an ambiguity, you can use the this prefix to identify in¬ 
stance variables). Understanding the distinctions among these three kinds of variables 
in instance methods is a key to success in object-oriented programming. 

instance 

public class Example variable 

j / 

private int var; 


local variable 


private void methodl() 

j 

i nt var; 

refers to local variable, NOT instance variable 

var . . . 

this.var 


i 


' refers to instance variable 


private void method2() 

j 

var 

} ' 


refers to instance variable 


Scope of instance and local variables in an instance method 
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API, clients, and implementations. These are the basic components that you need 
to understand to be able to build and use abstract data types in Java. Every ADT im¬ 
plementation that we will consider will be a Java class with private instance variables, 
constructors, instance methods, and a client. To fully understand a data type, we need 
the API, typical client code, and an implementation, summarized for Counter on the 
facing page. To emphasize the separation of client and implementation, we normally 
present each client as a separate class containing a static method main() and reserve 
test client’s mai n () in the data-type definition for minimal unit testing and develop¬ 
ment (calling each instance method at least once). In each data type that we develop, 
we go through the same steps. Rather than thinking about what action we need to take 
next to accomplish a computational goal (as we did when first learning to program), we 
think about the needs of a client, then accommodate them in an ADT, following these 
three steps: 

■ Specify an API. The purpose of the API is to separate clients from implementa¬ 
tions, to enable modular programming. We have two goals when specifying an 
API. First, we want to enable clear and correct client code. Indeed, it is a good 
idea to write some client code before finalizing the API to gain confidence that 
the specified data-type operations are the ones that clients need. Second, we 
want to be able to implement the operations. There is no point specifying opera¬ 
tions that we have no idea how to implement. 

■ Implement a Java class that meets the API specifications. First we choose the 
instance variables, then we write constructors and the instance methods. 

■ Develop multiple test clients, to validate the design decisions made in the first 
two steps. 

What operations do clients need to perform, and what data-type values can best sup¬ 
port those operations? These basic decisions are at the heart of every implementation 
that we develop. 
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API 


public class Counter 

Counter(String id) 
void incrementO 
int tallyO 

String toStringO 


create a counter named i d 
increment the counter 
number of increments since creation 
string representation 


typical client public class Flips 

{ 

public static void mai n(String[] args) 

{ 

int T = Integer.parselnt(args[0]); 

Counter heads = new Counter("heads"); 
Counter tails = new Counter("tai1s"); 

for (int t = 0; t < T; t++) 

if (StdRandom.bernoulli (0.5)) 
heads . i ncrementO ; 
else tai 1 s . i ncrementO ; 

StdOut.println(heads); 
StdOut.println(tails) ; 
int d = heads.tallyO - tails.tallyO ; 
StdOut.println("delta: " + Math.abs(d)); 

} 

} 


implementation 

public class Counter 

{ 

private final String name; 
private int count; 

public Counter(Stri ng id) 

{ name = id; } 

public void incrementO 
{ count++; } 

public int tallyO 
{ return count; } 

public String toStringO 
{ return count + " " + name; } 


application 

% java Flips 1000000 
500172 heads 
499828 tails 
delta: 344 


} 


An abstract data type for a simple counter 
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More ADT implementations As with any programming concept, the best way 
to understand the power and utility of ADTs is to consider carefully more examples 
and more implementations. There will be ample opportunity for you to do so, as much 
of this book is devoted to ADT implementations, but a few more simple examples will 
help us lay the groundwork for addressing them. 

Date. Shown on the facing page are two implementations of the Date ADT that we con¬ 
sidered on page 79. To reduce clutter, we omit the parsing constructor (which is described 
inEXERCiSE 1 . 2 . 19 )andtheinheritedmethodsequal s() (seepage 103),compareToO (see 
page 247), and hashCode () (see exercise 3 . 4 . 22 ). The straightforward implementation 
on the left maintains the day, month, and year as instance variables, so that the instance 
methods can just return the appropriate value; the more space-efficient implementa¬ 
tion on the right uses only a single i nt value to represent a date, using a mixed-radix 
number that represents the date with day d, month in, and year y as 512 y + 32 m + d. 
One way that a client might notice the difference between these implementations is by 
violating implicit assumptions: the second implementation depends for its correctness 
on the day being between 0 and 31, the month being between 0 and 15, and the year be¬ 
ing positive (in practice, both implementations should check that months are between 
1 and 12, days are between 1 and 31, and that dates such as June 31 and February 29, 
2009, are illegal, though that requires a bit more work). This example highlights the 
idea that we rarely fully specify implementation requirements in an API (we normally 
do the best we can, and could do better here). Another way that a client might notice the 
difference between the two implementations is performance: the implementation on the 
right uses less space to hold data-type values at the cost of more time to provide them to 
the client in the agreed form (one or two arithmetic operations are needed). Such trad¬ 
eoffs are common: one client may prefer one of the implementations and another client 
might prefer the other, so we need to accommodate both. Indeed, one of the recurring 
themes of this book is that we need to understand the space and time requirements of 
various implementations and their suitability for use by various clients. One of the key 
advantages of using data abstraction in our implementations is that we can normally 
change from one implementation to another without changing any client code. 

Maintaining multiple implementations. Multiple implementations of the same API 
can present maintainence and nomenclature issues. In some cases, we simply want to 
replace an old implementation with an improved one. In others, we may need to main¬ 
tain two implementations, one suitable for some clients, the other suitable for others. 
Indeed, a prime goal of this book is to consider in depth several implementations of 
each of a number of fundamental ADTs, generally with different performance charac¬ 
teristics. In this book, we often compare the performance of a single client using two 
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API public class Date 



Date(int month, int day, int year) 

create a date 

int 

month() 

month 

int 

day() 

day 

int 

year() 

year 

Stri ng 

toStri ng() 

string representation 


test client 

public static void mai n(String[] args) 

{ 

int m = Integer.parselnt(args[0]); 
int d = Integer.parselnt(args[1]); 
int y = Integer.parselnt(args[2]); 
Date date = new Date(m, d, y); 
StdOut.println(date) ; 

} 


application 

% java Date 12 31 1999 
12/31/1999 


implementation 

public class Date 

{ 

private final int month; 
private final int day; 
private final int year; 

public Date(int m, int d, int y) 

{ month = m; day = d; year = y; } 

public int month() 

{ return month; } 

public int day() 

{ return day; } 

public int year() 

{ return day; } 

public String toStringO 
{ return month() + "/" + day() 

+ "/" + yearQ; } 


alternate implementation 

public class Date 

{ 

private final int value; 

public Date(int m, int d, int y) 

{ value = y*512 + m*32 + d; } 

public int monthO 
{ return (value / 32) % 16; } 

public int day() 

{ return value % 32; } 

public int year() 

{ return value / 512; } 

public String toStringO 
{ return month() + "/" + day() 

+ "/" + yearQ; } 


An abstract data type to encapsulate dates, with two implementations 
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different implementations of the same API. For this reason, we generally adopt an in¬ 
formal naming convention where we: 

■ Identify different implementations of the same API by prepending a descrip¬ 
tive modifier. For example, we might name our Date implementations on the 
previous page Basi cDate and Smal 1 Date, and we might wish to develop a 
SmartDate implementation that can validate that dates are legal. 

■ Maintain a reference implementation with no prefix that makes a choice that 
should be suitable for most clients. That is, most clients should just use Date. 

In a large system, this solution is not ideal, as it might involve changing client code. For 
example, if we were to develop a new implementation ExtraSmal 1 Date, then our only 
options are to change client code or to make it the reference implementation for use by 
all clients, java has various advanced language mechanisms for maintaining multiple 
implementations without needing to change client code, but we use them sparingly 
because their use is challenging (and even controversial) even for experts, especially in 
conjuction with other advanced language features that we do value (generics and itera¬ 
tors). These issues are important (for example, ignoring them led to the celebrated Y2K 
problem at the turn of the millennium, because many programs used their own imple¬ 
mentations of the date abstraction that did not take into account the first two digits of 
the year), but detailed consideration of these issues would take us rather far afield from 
the study of algorithms. 

Accumulator. The accumulator API shown on the facing page defines an abstract data 
type that provides to clients the ability to maintain a running average of data values. For 
example, we use this data type frequently in this book to process experimental results 
(see section 1 . 4 ). The implementation is straightforward: it maintains a i nt instance 
variable counts the number of data values seen so far and a doubl e instance variable 
that keeps track of the sum of the values seen so far; to compute the average it divides 
the sum by the count. Note that the implementation does not save the data values—it 
could be used for a huge number of them (even on a device that is not capable of 
holding that many), or a huge number of accumulators could be used on a big system. 
This performance characteristic is subtle and might be specified in the API, because 
an implementation that does save the values might cause an application to run out of 
memory. 
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API 


public class Accumulator 

AccumulatorO 

void addDataValue(double val) 
double mean() 

String toStringO 


create an accumulator 
add a new data value 
mean of all data values 
string representation 


typical client public class TestAccumulator 

{ 

public static void main(String[] args) 

{ 

int T = Integer.parselnt(args[0]); 
Accumulator a = new AccumulatorO; 
for (int t = 0; t < T; t++) 

a.addDataValue(StdRandom. randomO); 
StdOut.println(a) ; 

} 


application % j ava TestAccumul ator 1000 
Mean (1000 values): 0.51829 

% java TestAccumulator 1000000 
Mean (1000000 values): 0.49948 

% java TestAccumulator 1000000 
Mean (1000000 values): 0.50014 


implementation public class Accumulator 

{ 

private double total; 
private int N; 

public void addDataVal ue(double val) 

{ 

N++; 

total += val; 

} 

public double mean() 

{ return total/N; } 

public String toStringO 
{ return "Mean (" + N + " values): " 

+ Stri ng. format("%7.5f" , meanQ); } 


An abstract data type for accumulating data values 
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Visual accumulator. The visual accumulator implementation shown on the facing 
page extends Accumulator to present a useful side effect: it draws on StdDraw all the 
data (in gray) and the running average (in red). 

The easiest way to do so is to add a constructor 
that provides the number of points to be plotted 
and the maximum value, for rescaling the plot. 

Vi sualAccumulator is not technically an imple¬ 
mentation of the Accumul ator API (its construc¬ 
tor has a different signature and it causes a differ¬ 
ent prescribed side effect). Generally, we are 
careful to fully specify APIs and are loath to make 
any changes in an API once articulated, as it might 
involve changing an unknown amount of client (and implementation) code, but add¬ 
ing a constructor to gain functionality can sometimes be defended because it involves 
changing the same line in client code that we change when changing a class name. In 
this example, if we have developed a client that uses an Accumul ator and perhaps has 
many calls to addDataValue() and avg(), we can enjoy the benefits of 
Vi sualAccumulator by just changing one line of client code. 


height of Nth red dot from the left 
is the av erage of the heights 



of the eftmost N gray dots 


height of gray dot 
is the data point value 


Visual accumulator plot 


application 



% java TestVisualAccumulator 2000 
Mean (2000 values): 0.509789 
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API public class Visual Accumulator 

VisualAccumulator^nt trials, double max) 
void addDataVal ue(doubl e val) add a new data value 
double avg() average of all data values 

String toStringO string representation 


typical client public class TestVi sual Accumul ator 

{ 

public static void main(String[] args) 

{ 

int T = Integer.parselnt(args[0]); 

Visual Accumulator a = new Visual Accumul ator(T, 1.0); 
for (int t = 0; t < T; t++) 

a.addDataValue(StdRandom. randomO) ; 
StdOut.println(a) ; 

} 

} 


implementation public class Vi sual Accumul ator 

{ 

private double total; 
private int N; 

public VisualAccumulator(int trials, double max) 

{ 

StdDraw.setXscale(0, trials); 
StdDraw.setYscale(0, max); 
StdDraw.setPenRadius(.005); 

} 

public void addDataVal ue(doubl e val) 

{ 

N++; 

total += val ; 

StdDraw.setPenColor(StdDraw.DARK„GRAY); 
StdDraw.point(N, val); 

StdDraw.setPenColor(StdDraw.RED); 

StdDraw.point(N, total/N); 

} 

public double mean() 
public String toStringO 
// Same as Accumulator. 


An abstract data type for accumulating data values (visual version) 
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Data-type design An abstract data type is a data type whose representation is hid¬ 
den from the client. This idea has had a powerful effect on modern programming. The 
various examples that we have considered give us the vocabulary to address advanced 
characteristics of ADTs and their implementation as Java classes. Many of these topics 
are, on the surface, tangential to the study of algorithms, so it is safe for you to skim 
this section and refer to it later in the context of specific implementation problems. Our 
goal is to put important information related to designing data types in one place for 
reference and to set the stage for implementations throughout this book. 

Encapsulation. A hallmark of object-oriented programming is that it enables us to 
encapsulate data types within their implementations, to facilitate separate development 
of clients and data type implementations. Encapsulation enables modular program¬ 
ming, allowing us to 

■ Independently develop of client and implementation code 

■ Substitute improved implementations without affecting clients 

■ Support programs not yet written (the API is a guide for any future client) 
Encapsulation also isolates data-type operations, which leads to the possibility of 

■ Limiting the potential for error 

■ Adding consistency checks and other debugging tools in implementations 

■ Clarifying client code 

An encapsulated data type can be used by any client, so it extends the Java language. 
The programming style that we are advocating is predicated on the idea of breaking 
large programs into small modules that can be developed and debugged independently. 
This approach improves the resiliency of our software by limiting and localizing the ef¬ 
fects of making changes, and it promotes code reuse by making it possible to substitute 
new implementations of a data type to improve performance, accuracy, or memory 
footprint. The same idea works in many settings. We often reap the benefits of encap¬ 
sulation when we use system libraries. New versions of the Java system often include 
new implementations of various data types or static method libraries, but the APIs do 
not change. In the context of the study of algorithms and data structures, there is strong 
and constant motivation to develop better algorithms because we can improve perfor¬ 
mance for all clients by substituting an improved ADT implementation without chang¬ 
ing the code of any client. The key to success in modular programming is to maintain 
independence among modules. We do so by insisting on the API being the only point of 
dependence between client and implementation. You do not need to know how a data 
type is implemented in order to use it and you can assume that a client knows nothing but 
the API when implementing a data type. Encapsulation is the key to attaining both of 
these advantages. 
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Designing APIs. One of the most important and most challenging steps in building 
modern software is designing APIs. This task takes practice, careful deliberation, and 
many iterations, but any time spent designing a good API is certain to be repaid in time 
saved debugging or code reuse. Articulating an API might seem to be overkill when 
writing a small program, but you should consider writing every program as though you 
will need to reuse the code someday. Ideally, an API would clearly articulate behavior 
for all possible inputs, including side effects, and then we would have software to check 
that implementations meet the specification. Unfortunately, a fundamental result from 
theoretical computer science known as the specification problem implies that this goal 
is actually impossible to achieve. Briefly, such a specification would have to be written 
in a formal language like a programming language, and the problem of determining 
whether two programs perform the same computation is known, mathematically, to be 
undecidable. Therefore, our APIs are brief English-language descriptions of the set of 
values in the associated abstract data type along with a list of constructors and instance 
methods, again with brief English-language descriptions of their purpose, including 
side effects. To validate the design, we always include examples of client code in the text 
surrounding our APIs. Within this broad outline, there are numerous pitfalls that every 
API design is susceptible to: 

■ An API may be too hard to implement, implying implementations that are dif¬ 
ficult or impossible to develop. 

■ An API may be too hard to use, leading to client code that is more complicated 
than it would be without the API. 

■ An API may be too narrow, omitting methods that clients need. 

■ An API may be too wide, including a large number of methods not needed 
by any client. This pitfall is perhaps the most common, and one of the most 
difficult to avoid. The size of an API tends to grow over time because it is not 
difficult to add methods to an existing API, but it is difficult to remove methods 
without breaking existing clients. 

■ An API may be too general, providing no useful abstractions. 

■ An API may be too specific, providing abstractions so detailed or so diffuse as to 
be useless. 

■ An API may be too dependent on a particular representation, therefore not serv¬ 
ing the purpose of freeing client code from the details of using that representa¬ 
tion. This pitfall is also difficult to avoid, because the representation is certainly 
central to the development of the implementation. 

These considerations are sometimes summarized in yet another motto: provide to cli¬ 
ents the methods they need and no others. 
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Algorithms and abstract data types. Data abstraction is naturally suited to the study 
of algorithms, because it helps us provide a framework within which we can precisely 
specify both what an algorithm needs to accomplish and how a client can make use of 
an algorithm. Typically, in this book, an algorithm is an implementation of an instance 
method in an abstract data type. For example, our whitelisting example at the begin¬ 
ning of the chapter is naturally cast as an ADT client, based on the following operations: 

■ Construct a SET from an array of given values. 

■ Determine whether a given value is in the set. 

These operations are encapsulated in the Stati cSEToflnts ADT, shown on the facing 
page along with Whi tel i st, a typical client. Stati cSEToflnts is a special case of the 
more general and more useful symbol table ADT that is the focus of chapter 3 . Binary 
search is one of several algorithms that we study that is suitable for implementing these 
ADTs. By comparison with the Bi narySearch implementation on page 47, this imple¬ 
mentation leads to clearer and more useful client code. For example, Stati cSEToflnts 
enforces the idea that the array must be sorted before rank() is called. With the abstract 
data type, we separate the client from the implementation making it easier for any client 
to benefit from the ingenuity of the binary search algorithm, just by following the API 
(clients of rank() in Bi narySearch have to know to sort the array first). Whitelisting is 
one of many clients that can take advantage of binary search. 

EVERY JAVA PROGRAM is a Set of 
static methods and/or a data type 
implementation. In this book, we 
focus primarily on abstract data 
type implementations such as 
Stati cSEToflnts, where the focus 
is on operations and the representa¬ 
tion of the data is hidden from the 
client. As this example illustrates, 
data abstraction enables us to 

■ Precisely specify what algorithms can provide for clients 

■ Separate algorithm implementations from the client code 

■ Develop layers of abstraction, where we make use of well-understood algorithms 
to develop other algorithms 

These are desirable properties of any approach to describing algorithms, whether it be 
an English-language description or pseudo-code. By embracing the Java cl ass mecha¬ 
nism in support of data abstraction, we have little to lose and much to gain: working 
code that we can test and use to compare performance for diverse clients. 


application 

% java Whitelist largeW.txt < largeT.txt 
499569 
984875 
295754 
207807 
140925 
161828 
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API public class StaticSEToflnts 

StaticSETofInts(int[] a) create a set from the values in a[] 
boolean contains(int key) is key in the set? 


typical client 


implementation 


Binary search recast as an object-oriented program (an ADT for search in a set of integers) 


public class Whitelist 

{ 

public static void mai n(String [] args) 

{ 

int[] w = In . readlnts(args [0]) ; 
StaticSEToflnts set = new StaticSEToflnts(w); 
while (!Stdln .isEmptyO) 

{ // Read key, print if not in whitelist. 

int key = Stdln.readlnt(); 
if (set.rank(key) == -1) 
StdOut.println(key) ; 

} 


} 


} 


import java.uti1.Arrays; 

public class StaticSEToflnts 

{ 

private int[] a; 

public StaticSETofInts(int[] keys) 

{ 

a = new int[keys.length]; 
for (int i =0; i < keys.length; i++) 
a[i] = keys(i] ; // defensive copy 
Arrays.sort(a); 

} 

public boolean contains(int key) 

{ return rank(key) != -1; } 

private int rank(int key) 

{ // Binary search, 

i nt 1 o = 0; 
int hi = a.length - 1; 
while (lo <= hi) 

{ // Key is in a[lo..hi] or not present. 


int mid = lo 
if (key 

else if (key 
el se 

} 

return -1; 


+ (hi - lo) / 2; 

< a[mid]) hi = mid - 
> a[mid]) lo = mid + 
return mid; 
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Interface inheritance. Java provides language support for defining relationships 
among objects, known as inheritance. These mechanisms are widely used by software 
developers, so you will study them in detail if you take a course in software engineer¬ 
ing. The first inheritance mechanism that we consider is known as subtyping, which 
allows us to specify a relationship between otherwise unrelated classes by specifying in 
an interface a set of common methods that each implementing class must contain. An 
interface is nothing more than a list of instance methods. For example, instead of using 
our informal API, we might have articulated an interface for Date: 

public interface Datable 

{ 

int monthO ; 
i nt dayO ; 
int year(); 

} 

and then referred to the interface in our implementation code 

public class Date implements Datable 

{ 

// implementation code (same as before) 

} 

so that the Java compiler will check that it matches the interface. Adding the code 
implements Datable to any class that implements month(), day(), and year() pro¬ 
vides a guarantee to any client that an object of that class can invoke those methods. 
This arrangement is known as interface inheritance —an implementing class inherits the 
interface. Interface inheritance allows us to write client programs that can manipulate 


interface 

methods 

section 

objects of any type that implements 
the interface (even a type to be creat- 

java.1ang.Comparable 

compareToO 

2.1 

ed in the future), by invoking meth- 

comparison 

java.util.Comparator 

compareO 

2.5 

ods in the interface. We might have 
used interface inheritance in place of 

java, lang. Iterabl e 

i teratorO 

1-3 

our more informal APIs, but chose 
not to do so to avoid dependence on 
specific high-level language mecha- 

iteration 

hasNextO 


nisms that are not critical to the 
understanding of algorithms and 

java.util.Iterator 

next() 

1-3 

to avoid the extra baggage of inter- 

remove (j 

Java interfaces used in this book 


face files. But there are a few situa¬ 
tions where Java conventions make 
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it worthwhile for us to take advantage of interfaces: we use them for comparison and for 
iteration, as detailed in the table at the bottom of the previous page, and will consider 
them in more detail when we cover those concepts. 

Implementation inheritance. Java also supports another inheritence mechanism 
known as subclassing, which is a powerful technique that enables a programmer to 
change behavior and add functionality without rewriting an entire class from scratch. 
The idea is to define a new class ( subclass, or derived class) that inherits instance meth¬ 
ods and instance variables from another class ( superclass, or base class). The subclass 
contains more methods than the superclass. Moreover, the subclass can redefine or 
override methods in the superclass. Subclassing is widely used by systems programmers 
to build so-called extensible libraries—one programmer (even you) can add methods to 
a library built by another programmer (or, perhaps, a team of systems programmers), 
effectively reusing the code in a potentially huge library. For example, this approach is 
widely used in the development of graphical user interfaces, so that the large amount of 
code required to provide all the facilities that users expect (drop-down menus, cut-and- 
paste, access to files, and so forth) can be reused. The use of subclassing is controversial 
among systems and applications programmers (its advantages over interface inheri¬ 
tance are debatable), and we avoid it in this book because it generally works against 
encapsulation. Certain vestiges of the approach are built in to Java and therefore un¬ 
avoidable: specifically, every class is a subtype of Java’s Object class. This structure 
enables the “convention” that every class includes an implementation of getCl ass(), 
toStri ng(), equal s(), hashCodeO, and several other methods that we do not use in 
this book. Actually, every class inherits these methods from Object through subclassing, 
so any client can use them for any object. We usually override toStri ng(), equal s (), 
hashCodeO in new classes because the default Object implementation generally does 
not lead to the desired behavior. We now will consider toStri ng() and equal s(); we 
discuss hashCodeO in section 3 . 4 . 



method 

purpose 

section 

Class 

getClassO 

what class is this object? 

1.2 

Stri ng 

toStri ng() 

string representation of this object 

1.1 

boolean 

equals(0bject that) 

is this object equal to that? 

1.2 

i nt 

hashCodeO 

hash code for this object 

3-4 


Inherited methods from Object used in this book 
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String conversion. By convention, every Java type inherits toSt ri ng () from Object, 
so any client can invoke toStri ng() for any object. This convention is the basis for Ja¬ 
va’s automatic conversion of one operand of the concatenation operator + to a Stri ng 
whenever the other operand is a Stri ng. If an object’s data type does not include an 
implementation of toSt ri ng (), then the default implementation in Ob j ect is invoked, 
which is normally not helpful, since it typically returns a string representation of the 
memory address of the object. Accordingly, we generally include implementations of 
toStri ng() that override the default in every class that we develop, as highlighted for 
Date on the facing page. As illustrated in this code, toStri ng() implementations are 
often quite simple, implicitly (through +) using toStri ng() for each instance variable. 

Wrapper types. Java supplies built-in reference types known as wrapper types, one for 
each of the primitive types: Bool ean, Byte, Character, Doubl e, Float, Integer, Long, 
and Short correspond to boolean, byte, char, double, float, int, long, and short, 
respectively. These classes consist primarily of static methods such as parselntO but 
they also include the inherited instance methods toStringO, compa r eT o (), equal s (), 
and hashCode(). Java automatically converts from primitive types to wrapper types 
when warranted, as described on page 122. For example, when an i nt value is concat¬ 
enated with a Stri ng, it is converted to an Integer that can invoke toStri ng(). 

Equality. What does it mean for two objects to be equal? If we test equality with 
(a == b) where a and b are reference variables of the same type, we are testing whether 
they have the same identity: whether the references are equal. Typical clients would 
rather be able to test whether the data-type values (object state) are the same, or to 
implement some type-specific rule. Java gives us a head start by providing implementa¬ 
tions both for standard types such as Integer, Doubl e, and Stri ng and for more com¬ 
plicated types such as Fi 1 e and URL. When using these types of data, you can just use the 
built-in implementation. For example, if x and y are Stri ng values, then x. equal s (y) 
is true if and only if x and y have the same length and are identical in each character 
position. When we define our own data types, such as Date or Transacti on, we need 
to override equal s(). Java’s convention is that equal sCD must be an equivalence rela¬ 
tion. It must be 

■ Reflexive: x. equal s(x) is true. 

■ Symmetric: x. equal s(y) is true if and only if y. equal s(x) . 

■ Transitive: if x. equal s(y) and y. equal s(z) are true, then so is x. equal s(z) . 
In addition, it must take an Ob j ect as argument and satisfy the following properties. 

■ Consistent: multiple invocations of x. equal s (y) consistently return the same 
value, provided neither object is modified. 

■ Not null: x. equals (null) returns false. 
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These are natural definitions, but ensuring that these properties hold, adhering to Java 
conventions, and avoiding unnecessary work in an implementation can be tricky, as il¬ 
lustrated for Date below. It takes the following step-by-step approach: 

■ If the reference to this object is the same as the reference to the argument object, 
return true. This test saves the work of doing all the other checks in this case. 

■ If the argument is null, return fal se, to adhere to the convention (and to avoid 
following a null reference in code to follow). 

■ If the objects are not from the same class, return fal se. To determine an object’s 
class, we use getCl ass O. Note that we can use == to tell us whether two objects 
of type Cl ass are equal because getCl ass () is guaranteed to return the same 
reference for all objects in any given class. 

■ Cast the argument 
from Ob j ect to Date 
(this cast must succeed 
because of the previous 
test). 

■ Return fal se if any 
instance variables do 
not match. For other 
classes, some other 
definition of equality 
might be appropriate. 

For example, we might 
regard two Counter 
objects as equal if their 
count instance variables 
are equal. 

This implementation is a 
model that you can use to 
implement equal s() for any 
type that you implement. 

Once you have implemented 
one equal s(), you will not 
find it difficult to implement 
another. 


Overriding toStringO and equalsQ in a data type definition 


public class Date 
{ 

private final int month; 
private final int day; 
private final int year; 

public Date(int m, int d, int y) 

{ month = m; day = d; year = y; } 

public int monthO 
{ return month; } 

public int dayO 
{ return day; } 

public int year() 

{ return year; } 

public String toStringO 
{ return monthO + "/" + day() 


"/" + year() ; } 


public boolean equals(0bject x) 

{ 

if (this == x) return true; 
if (x == null) return false; 
if (this.getClassO != x.getClassO) 
Date that = (Date) x; 
if (this.day != that.day) 
if (this.month != that.month) 
if (this.year != that.year) 
return true; 

} 


return 

fal se 

return 

fal se 

return 

fal se 

return 

fal se 
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Date a 
Date b 
b = a; 


655 

656 

657 


811 

812 

813 


new Date(12 
new Date(l, 


31, 1999); 
1 . 2011 ); 


811 


811 


12 


31 


1999 


. references to 
' same object 


orphaned 

object 


Memory management. The ability to assign a new value to a reference variable cre¬ 
ates the possibility that a program may have created an object that can no longer be 
referenced. For example, consider the three assignment statements in the figure at left. 
After the third assignment statement, not only do a and b refer to the same Date object 
(1/1/2011), but also there is no longer a reference to the Date object that was created 

and used to initialize b. The only reference to that object 
was in the variable b, and this reference was overwritten 
by the assignment, so there is no way to refer to the object 
again. Such an object is said to be orphaned. Objects are 
also orphaned when they go out of scope. Java programs 
tend to create huge numbers of objects (and variables that 
hold primitive data-type values), but only have a need for a 
small number of them at any given point in time. Accord¬ 
ingly, programming languages and systems need mecha¬ 
nisms to allocate memory for data-type values during the 
time they are needed and to free the memory when they 
are no longer needed (for an object, sometime after it is 
orphaned). Memory management turns out to be easier 
for primitive types because all of the information needed 
for memory allocation is known at compile time. Java (and 
most other systems) takes care of reserving space for vari¬ 
ables when they are declared and freeing that space when 
they go out of scope. Memory management for objects is 
more complicated: the system can allocate memory for an 
object when it is created, but cannot know precisely when 
to free the memory associated with each object because 
the dynamics of a program in execution determines when 
objects are orphaned. In many languages (such as C and 
C++) the programmer is responsible for both allocating 
and freeing memory. Doing so is tedious and notoriously 
error-prone. One of Java’s most significant features is its ability to automatically man¬ 
age memory. The idea is to free the programmers from the responsibility of managing 
memory by keeping track of orphaned objects and returning the memory they use to 
a pool of free memory. Reclaiming memory in this way is known as garbage collection. 
One of Java’s characteristic features is its policy that references cannot be modified. 
This policy enables Java to do efficient automatic garbage collection. Programmers still 
debate whether the overhead of automatic garbage collection justifies the convenience 
of not having to worry about memory management. 


2011 


- New Year's 
Eve 1999 


New Years 
' Day 2011 


An orphaned object 
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mutable 


Counter 


Immutability. An immutable data type, such as Date, has the property that the value 
of an object never changes once constructed. By contrast, a mutable data type, such as 
Counter or Accumul ator, manipulates object values that are intended to change. Java’s 
language support for helping to enforce immutability is the f i nal modifier. When you 
declare a variable to be fi nal, you are promising to assign it a value only once, either 
in an initializer or in the constructor. Code that could modify the value of a fi nal 
variable leads to a compile-time error. In our code, we use the modifier fi nal with 
instance variables whose values never change. This policy serves as documentation that 
the value does not change, prevents accidental changes, and makes programs easier 
to debug. For example, you do not have to include a fi nal value in a trace, since you 
know that its value never changes. A data type such as Date whose instance variables 
are all primitive and fi nal is immutable (in code that does not use implementation 
inheritence, our convention). Whether to make a data type immutable is an important 
design decision and depends on the application at hand. For data 
types such as Date, the purpose of the abstraction is to encap¬ 
sulate values that do not change so that we can use them in as¬ 
signment statements and as arguments and return values from 
functions in the same way as we use primitive types (without hav¬ 
ing to worry about their values changing). A programmer imple- J ava arrays 
menting a Date client might reasonably expect to write the code 
d = dO for two Date variables, in the same way as for doubl e or 
i nt values. But if Date were mutable and the value of d were to 
change after the assignment d = dO, then the value of dO would also change (they are 
both references to the same object)! On the other hand, for data types such as Counter 
and Accumulator, the very purpose of the abstraction is to encapsulate values as they 
change. You have already encountered this distinction as a client programmer, when 
using Java arrays (mutable) and Java’s Stri ng data type (immutable). When you pass 
a Stri ng to a method, you do not worry about that method changing the sequence of 
characters in the Stri ng,but when you pass an array to a method, the method is free to 
change the contents of the array. Stri ng objects are immutable because we generally do 
not want Stri ng values to change, and Java arrays are mutable because we generally do 
want array values to change. There are also situations where we want to have mutable 
strings (that is the purpose of Java’s Stri ngBui 1 der class) and where we want to have 
immutable arrays (that is the purpose of the Vector class that we consider later in this 
section). Generally, immutable types are easier to use and harder to misuse than muta¬ 
ble types because the scope of code that can change their values is far smaller. It is easier 
to debug code that uses immutable types because it is easier to guarantee that variables 
in client code that uses them remain in a consistent state. When using mutable types, 


immutable 


Date 


Stri ng 


Mutable/immutable examples 
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you must always be concerned about where and when their values change. The down¬ 
side of immutability is that a new object must be created for every value. This expense is 
normally manageable because Java garbage collectors are typically optimized for such 
situations. Another downside of immutability stems from the fact that, unfortunately, 
fi nal guarantees immutability only when instance variables are primitive types, not 
reference types. If an instance variable of a reference type has the fi nal modifier, the 
value of that instance variable (the reference to an object) will never change—it will 
always refer to the same object—but the value of the object itself can change. For ex¬ 
ample, this code does not implement an immutable type: 

public class Vector 

{ 

private final doublet] coords; 

public Vector(double[] a) 

{ coords = a; } 

} 

A client program could create a Vector by specifying the entries in an array, and then 
(bypassing the API) change the elements of the Vector after construction: 

doublet] a = { 3.0, 4.0 }; 

Vector vector = new Vector(a); 

afO] =0.0; // Bypasses the public API. 

The instance variable coords t] is pri vate and fi nal , but Vector is mutable because 
the client holds a reference to the data. Immutability needs to be taken into account in 
any data-type design, and whether a data type is immutable should be specified in the 
API, so that clients know that object values will not change. In this book, our primary 
interest in immutability is for use in certifying the correctness of our algorithms. For 
example, if the type of data used for a binary search algorithm were mutable, then cli¬ 
ents could invalidate our assumption that the array is sorted for binary search. 
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Design by contract. To conclude, we briefly discuss Java language mechanisms that 
enables you to verify assumptions about your program as it is running. We use two Java 
language mechanisms for this purpose: 

■ Exceptions, which generally handle unforeseen errors outside our control 

■ Assertions, which verify assumptions that we make within code we develop 
Liberal use of both exceptions and assertions is good programming practice. We use 
them sparingly in the book for economy, but you will find them throughout the code 
on the booksite. This code aligns with a substantial amount of the surrounding com¬ 
mentary about each algorithm in the text that has to do with exceptional conditions 
and with asserted invariants. 

Exceptions and errors. Exceptions and errors are disruptive events that occur while a 
program is running, often to signal an error. The action taken is known as throwing an 
exception or throwing an error. We have already encountered exceptions thrown by Java 
system methods in the course of learning basic features of Java: StackOverfi owError, 
ArithmeticException, ArraylndexOutOfBoundsException, OutOfMemoryError, 
and Nul 1 Poi nterException are typical examples. You can also create your own ex¬ 
ceptions. The simplest kind is a RuntimeException that terminates execution of the 
program and prints an error message 

throw new RuntimeExceptionC'Error message here."); 

A general practice known as fail fast programming suggests that an error is more easily 
pinpointed if an exception is thrown as soon as an error is discovered (as opposed to 
ignoring the error and deferring the exception to sometime in the future). 

Assertions. An assertion is a boolean expression that you are affirming is true at that 
point in the program. If the expression is fal se, the program will terminate and re¬ 
port an error message. We use assertions both to gain confidence in the correctness of 
programs and to document intent. For example, suppose that you have a computed 
value that you might use to index into an array. If this value were negative, it would 
cause an Ar raylndexOutOfBoundsExcepti on sometime later. But if you write the code 
assert index >= 0; you can pinpoint the place where the error occurred. You can 
also add an optional detail message such as 

assert index >= 0 : "Negative index in method X"; 

to help you locate the bug. By default, assertions are disabled. You can enable them from 
the command line by using the -enabl easserti ons flag (-ea for short). Assertions are 
for debugging: your program should not rely on assertions for normal operation since 
they may be disabled. When you take a course in systems programming, you will learn 
to use assertions to ensure that your code never terminates in a system error or goes into 
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an infinite loop. One model, known as the design-by-contract model of programming 
expresses the idea. The designer of a data type expresses a precondition (the condition 
that the client promises to satisfy when calling a method), a postcondition (the condi¬ 
tion that the implementation promises to achieve when returning from a method), and 
side effects (any other change in state that the method could cause). During develop¬ 
ment, these conditions can be tested with assertions. 

Summary. The language mechanisms discussed throughout this section illustrate that 
effective data-type design leads to nontrivial issues that are not easy to resolve. Ex¬ 
perts are still debating the best ways to support some of the design ideas that we are 
discussing. Why does Java not allow functions as arguments? Why does Matlab copy 
arrays passed as arguments to functions? As mentioned early in chapter i, it is a slip¬ 
pery slope from complaining about features in a programming language to becoming 
a programming-language designer. If you do not plan to do so, your best strategy is 
to use widely available languages. Most systems have extensive libraries that you cer¬ 
tainly should use when appropriate, but you often can simplify your client code and 
protect yourself by building abstractions that can easily transport to other languages. 
Your main goal is to develop data types so that most of your work is done at a level of 
abstraction that is appropriate to the problem at hand. 

The table on the facing page summarizes the various kinds of Java classes that we 
have considered. 
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kind of class 


static methods 


immutable 
abstract data type 


mutable 

abstract data type 

abstract data type 
with I/O side effects 


examples 


characteristics 


Math Stdln StdOut 


no instance variables 


instance variables all p ri vate 

Date Transacti on instance variables all fi nal 

String Integer defensive copy for reference types 

Note: these are necessary but not sufficient. 


Counter Accumulator 


instance variables all pri vate 
not all instance variables fi nal 


Vi sual Accumulator instance variables all private 

In Out Draw instance methods do I/O 

Java classes (data-type implementations) 
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| Q&A 

Q. Why bother with data abstraction? 

A. It helps us produce reliable and correct code. For example, in the 2000 presidential 
election, A1 Gore received -16,022 votes on an electronic voting machine in Volusia 
County, Florida—the tally was clearly not properly encapsulated in the voting machine 
software! 

Q. Why the distinction between primitive and reference types? Why not just have refer¬ 
ence types? 

A. Performance. Java provides the reference types Integer, Doubl e, and so forth that 
correspond to primitive types that can be used by programmers who prefer to ignore 
the distinction. Primitive types are closer to the types of data that are supported by 
computer hardware, so programs that use them usually run faster than programs that 
use corresponding reference types. 

Q. Do data types have to be abstract? 

A. No. Java also allows public and protected to allow some clients to refer directly 
to instance variables. As described in the text, the advantages of allowing client code to 
directly refer to data are greatly outweighed by the disadvantages of dependence on a 
particular representation, so all instance variables are pri vate in our code. We also oc¬ 
casionally use pri vate instance methods to share code among public methods. 

Q. What happens if I forget to use new when creating an object? 

A. To Java, it looks as though you want to call a static method with a return value of the 
object type. Since you have not defined such a method, the error message is the same as 
anytime you refer to an undefined symbol. If you compile the code 

Counter c = Counter("test"); 

you get this error message: 

cannot find symbol 

symbol : method Counter(String) 

You get the same kind of error message if you provide the wrong number of arguments 
to a constructor. 
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Q. What happens if I forget to use new when creating an array of objects? 

A. You need to use new for each object that you create, so when you create an array of 
N objects, you need to use new N+ 1 times: once for the array and once for each of the 
objects. If you forget to create the array: 

Counter[] a; 

a[0] = new Counter("test"); 

you get the same error message that you would get when trying to assign a value to any 
uninitialized variable: 

variable a might not have been initialized 
a[0] = new Counter("test"); 

A 

but if you forget to use new when creating an object within the array and then try to use 
it to invoke a method: 

Counter[] a = new Counter[2]; 
a[0] . i ncrementO ; 

you get a NullPointerException. 

Q. Why not write StdOut. pri ntl n(x . toStri ng()) to print objects? 

A. That code works fine, but java saves us the trouble of writing it by automatically 
invoking the toStri ng() method for any object, since pri ntl n() has a method that 
takes an Ob j ect as argument. 

Q. What is a pointer ? 

A. Good question. Perhaps that should be Nul 1 ReferenceExcepti on. Like a Java ref¬ 
erence, you can think of a pointer as a machine address. In many programming lan¬ 
guages, the pointer is a primitive data type that programmers can manipulate in many 
ways. But programming with pointers is notoriously error-prone, so operations pro¬ 
vided for pointers need to be carefully designed to help programmers avoid errors. 
Java takes this point of view to an extreme (that is favored by many modern program¬ 
ming-language designers). In Java, there is only oneway to create a reference (new) and 
only one way to change a reference (with an assignment statement). That is, the only 
things that a programmer can do with references are to create them and copy them. In 
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Q & A (continued) 


programming-language jargon, Java references are known as safe pointers, because Java 
can guarantee that each reference points to an object of the specified type (and it can 
determine which objects are not in use, for garbage collection). Programmers used to 
writing code that directly manipulates pointers think of Java as having no pointers at 
all, but people still debate whether it is really desirable to have unsafe pointers. 

Q. Where can I find more details on how Java implements references and does garbage 
collection? 

A. One Java system might differ completely from another. For example, one natural 
scheme is to use a pointer (machine address); another is to use a handle (a pointer to 
a pointer). The former gives faster access to data; the latter provides for better garbage 
collection. 

Q. What exactly does it mean to i mport a name? 

A. Not much: it just saves some typing. You could type java, uti 1 .Arrays instead of 
Arrays everywhere in your code instead of using the i mport statement. 

Q. What is the problem with implementation inheritance? 

A. Subtyping makes modular programming more difficult for two reasons. First, any 
change in the superclass affects all subclasses. The subclass cannot be developed inde¬ 
pendently of the superclass; indeed, it is completely dependent on the superclass. This 
problem is known as the fragile base class problem. Second, the subclass code, hav¬ 
ing access to instance variables, can subvert the intention of the superclass code. For 
example, the designer of a class like Counter for a voting system may take great care 
to make it so that Counter can only increment the tally by one (remember A1 Gore’s 
problem). But a subclass, with full access to the instance variable, can change it to any 
value whatever. 

Q. How do I make a class immutable? 

A. To ensure immutability of a data type that includes an instance variable of a mu¬ 
table type, we need to make a local copy, known as a defensive copy. And that may not be 
enough. Making the copy is one challenge; ensuring that none of the instance methods 
change values is another. 

Q. What is null? 
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A. It is a literal value that refers to no object. Invoking a method using the null ref¬ 
erence is meaningless and results in a Null PointerException. If you get this error 
message, check to make sure that your constructor properly initializes all of its instance 
variables. 

Q. Can I have a static method in a class that implements a data type? 

A. Of course. For example, all of our classes have mai n () . Also, it is natural to consider 
adding static methods for operations that involve multiple objects where none of them 
naturally suggests itself as the one that should invoke the method. For example, we 
might define a static method like the following within Poi nt: 

public static double distance(Point a, Point b) 

{ 

return a.distTo(b); 

} 

Often, including such methods can serve to clarify client code. 

Q. Are there other kinds of variables besides parameter, local, and instance variables? 

A. If you include the keyword stati c in a class declaration (outside of any type) it 
creates a completely different type of variable, known as a static variable. Like instance 
variables, static variables are accessible to every method in the class; however, they are 
not associated with any object. In older programming languages, such variables are 
known as global variables, because of their global scope. In modern programming, we 
focus on limiting scope and therefore rarely use such variables. When we do, we will call 
attention to them. 

Q. What is a deprecated method? 

A. A method that is no longer fully supported, but kept in an API to maintain compat¬ 
ibility. For example, Java once included a method Character.isSpaceO, and pro¬ 
grammers wrote programs that relied on using that method’s behavior. When the de¬ 
signers of Java later wanted to support additional Unicode whitespace characters, they 
could not change the behavior of isSpaceO without breaking client programs, so, 
instead, they added a new method, Character, isWhiteSpaceO, and deprecated the 
old method. As time wears on, this practice certainly complicates APIs. Sometimes, en¬ 
tire classes are deprecated. For example, Java deprecated its j ava .util . Date in order 
to better support internationalization. 
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EXERCISES 

1 . 2.1 Write a Point2D client that takes an integer value N from the command line, 
generates N random points in the unit square, and computes the distance separating 
the closest pair of points. 

1 . 2.2 Write an Interval ID client that takes an int value N as command-line argu¬ 
ment, reads N intervals (each defined by a pair of doubl e values) from standard input, 
and prints all pairs that intersect. 

1 . 2.3 Write an Interval 2D client that takes command-line arguments N, mi n, and max 
and generates N random 2D intervals whose width and height are uniformly distributed 
between mi n and max in the unit square. Draw them on StdDraw and print the number 
of pairs of intervals that intersect and the number of intervals that are contained in one 
another. 

1 . 2.4 What does the following code fragment print? 

String stringl = "hello"; 

String string2 = stringl; 
stringl = "world"; 

StdOut.println(stringl); 

Std0ut.println(string2); 

1 . 2.5 What does the following code fragment print? 

String s = "Hello World"; 
s. toUpperCaseO ; 
s.substring(6, 11); 

StdOut.println(s); 

Answer : "Hello World". String objects are immutable—string methods return 
a new String object with the appropriate value (but they do not change the value 
of the object that was used to invoke them). This code ignores the objects returned 
and just prints the original string. To print "WORLD", use s = s. toUpperCaseO and 
s = s.substring(6, 11). 

1 . 2.6 A string s is a circular rotation of a string t if it matches when the characters 
are circularly shifted by any number of positions; e.g., ACTGACG is a circular shift of 
TGACGAC, and vice versa. Detecting this condition is important in the study of genomic 
sequences. Write a program that checks whether two given strings s and t are circular 


www.it-ebooks.info 




1.2 ■ Data Abstraction 


115 


shifts of one another. Hint: The solution is a one-liner with i ndexOf (), 1 ength (), and 
string concatenation. 

1 . 2.7 What does the following recursive function return? 

public static String mysteryCString s) 

{ 

int N = s.1engthf); 
if (N <= 1) return s; 

String a = s.substring(0, N/2); 

String b = s.substring(N/2, N); 
return mystery(b) + mystery(a); 

} 

1 . 2.8 Suppose that a[] and b[] are each integer arrays consisting of millions of inte¬ 
gers. What does the follow code do? Is it reasonably efficient? 

int[] t = a; a=b; b=t; 

Answer. It swaps them. It could hardly be more efficient because it does so by copying 
references, so that it is not necessary to copy millions of elements. 

1 . 2.9 Instrument Bi narySearch (page 47) to use a Counter to count the total number 
of keys examined during all searches and then print the total after all searches are com¬ 
plete. Hint: Create a Counter in mai n() and pass it as an argument to rank(). 

1 . 2.10 Develop a class VisualCounter that allows both increment and decrement 
operations. Take two arguments N and max in the constructor, where N specifies the 
maximum number of operations and max specifies the maximum absolute value for 
the counter. As a side effect, create a plot showing the value of the counter each time its 
tally changes. 

1 . 2.11 Develop an implementation SmartDate of our Date API that raises an excep¬ 
tion if the date is not legal. 

1 . 2.12 Add a method dayOfTheWeekO to SmartDate that returns a String value 
Monday, Tuesday, Wednesday, Thursday, Fri day, Saturday, or Sunday, giving the ap¬ 
propriate day of the week for the date. You may assume that the date is in the 21st 
century. 
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EXERCISES (continued) 


1 . 2.13 Using our implementation of Date as a model (page 91), develop an implementa¬ 
tion of T ransaction. 

1 . 2.14 Using our implementation of equals() in Date as a model (page 103), develop 
implementations of equal s () for Transacti on. 
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CREATIVE PROBLEMS 

1.2.15 File input. Develop a possible implementation of the static readlnts Q meth¬ 
od from In (which we use for various test clients, such as binary search on page 47) that 
is based on the spl i t() method in Stri ng. 

Solution: 

public static int[] readlntsfString name) 

{ 

In in = new In(name); 

String input = Stdln.readAll(); 

String[] words = input.split("\\s+"); 
int[] ints = new int[words.1ength; 
for int i =0; i < word.length; i++) 
ints[i] = Integer.parseInt(words[i]); 
return ints; 

} 

We will consider a different implementation in section 1.3 (see page 126). 

1.2.16 Rational numbers. Implement an immutable data type Rational for rational 
numbers that supports addition, subtraction, multiplication, and division. 


public class Rational 



Rational (int numerator. 

int denominator) 

Rational 

pi us(Rational b) 

sum of this number and b 

Rational 

minus(Rational b) 

difference of this number and b 

Rational 

times (Rational b) 

product of this number and b 

Rational 

divides(Rational b) 

quotient of this number and b 

boolean 

equals(Rational that) 

is this number equal to that ? 

Stri ng 

toStri ng() 

string representation 


You do not have to worry about testing for overflow (see exercise 1 . 2 . 17 ), but use as 
instance variables two long values that represent the numerator and denominator to 
limit the possibility of overflow. Use Euclid’s algorithm (see page 4) to ensure that the 
numerator and denominator never have any common factors. Include a test client that 
exercises all of your methods. 
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CREATIVE PROBLEMS (continued) 


1.2.17 Robust implementation of rational numbers. Use assertions to develop an im¬ 
plementation of Rati onal (see exercise 1 . 2 . 16 ) that is immune to overflow. 

1.2.18 Variance for accumulator. Validate that the following code, which adds the 
methods var () and stddevO to Accumulator, computes both the mean and variance 
of the numbers presented as arguments to addDataVal ue(): 

public class Accumulator 

{ 

private double m; 
private double s; 
private int N; 

public void addDataVal uefdouble x) 

{ 

N++; 

s = s + 1.0 * (N-l) / N * (x - m) * (x - m) ; 
m = m+ (x-m) / N; 

} 

public double mean() 

{ return m; } 

public double var() 

{ return s/(N - 1); } 

public double stddevO 
{ return Math.sqrtfthis.var()); } 

} 

This implementation is less susceptible to roundoff error than the straightforward im¬ 
plementation based on saving the sum of the squares of the numbers. 
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1 . 2.19 Parsing. Develop the parse constructors for your Date and Transacti on im¬ 
plementations of exercise 1.2.13 that take a single String argument to specify the 
initialization values, using the formats given in the table below. 

Partial solution: 

public Date(String date) 

{ 

String[] fields = date.split("/"); 
month = Integer.parselnt(fields[0]); 
day = Integer.parselnt(fields[1]); 
year = Integer.parselnt(fields[2]); 

} 


type 


format 


example 


Date integers separated by slashes 


5/22/1939 


T customer, date, and amount, _ . c/n/tmn ,, nn 

Transaction Turing 5/22/1939 11.99 

separated by whitespace 

Formats for parsing 
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several fundamental data types involve collections of objects. Specifically, the set 
of values is a collection of objects, and the operations revolve around adding, remov¬ 
ing, or examining objects in the collection. In this section, we consider three such data 
types, known as the bag, the queue, and the stack. They differ in the specification of 
which object is to be removed or examined next. 

Bags, queues, and stacks are fundamental and broadly useful. We use them in imple¬ 
mentations throughout the book. Beyond this direct applicability, the client and imple¬ 
mentation code in this section serves as an introduction to our general approach to the 
development of data structures and algorithms. 

One goal of this section is to emphasize the idea that the way in which we represent 
the objects in the collection directly impacts the efficiency of the various operations. 
For collections, we design data structures for representing the collection of objects that 
can support efficient implementation of the requisite operations. 

A second goal of this section is to introduce generics and iteration, basic Java con¬ 
structs that substantially simplify client code. These are advanced programming-lan¬ 
guage mechanisms that are not necessarily essential to the understanding of algorithms, 
but their use allows us to develop client code (and implementations of algorithms) that 
is more clear, compact, and elegant than would otherwise be possible. 

A third goal of this section is to introduce and show the importance of linked data 
structures. In particular, a classic data structure known as the linked list enables im¬ 
plementation of bags, queues, and stacks that achieve efficiencies not otherwise pos¬ 
sible. Understanding linked lists is a key first step to the study of algorithms and data 
structures. 

For each of the three types, we consider APIs and sample client programs, then 
look at possible representations of the data type values and implementations of the 
data-type operations. This scenario repeats (with more complicated data structures) 
throughout this book. The implementations here are models of implementations later 
in the book and worthy of careful study. 


120 


www.it-ebooks.info 



1.3 ■ Bags, Queues, and Stacks 


121 


APIs As usual, we begin our discussion of abstract data types for collections by de¬ 
fining their APIs, shown below. Each contains a no-argument constructor, a method to 
add an item to the collection, a method to test whether the collection is empty, and a 
method that returns the size of the collection. Stack and Queue each have a method to 
remove a particular item from the collection. Beyond these basics, these APIs reflect two 
Java features that we will describe on the next few pages: generics and iterable collections. 


Bag 


public class Bag<Item> implements Iterable<Item> 


BagO 

void addfltem item) 
boolean isEmptyO 
int sizeQ 


create an empty bag 
add an item 
is the bag empty? 
number of items in the bag 


FIFO queue 

public class Queue<Item> implements Iterable<Item> 



QueueO 

create an empty queue 

voi d 

enqueue(Item item) 

add an item 

Item 

dequeueO 

remove the least recently added item 

boolean 

i sEmptyO 

is the queue empty? 

int 

sizeQ 

number of items in the queue 

)wn (UFO) stack 



public class 

Stack<Item> implements Iterable<Item> 


Stack() 

create an empty stack 

voi d 

pushfltem item) 

add an item 

Item 

pop() 

remove the most recently added item 

boolean 

i sEmptyO 

is the stack empty? 

int 

sizeQ 

number of items in the stack 


APIs for fundamental generic iterable collections 
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Generics. An essential characteristic of collection ADTs is that we should be able to use 
them for any type of data. A specific Java mechanism known as generics, also known 
as parameterized types, enables this capability. The impact of generics on the program¬ 
ming language is sufficiently deep that they are not found in many languages (including 
early versions of Java), but our use of them in the present context involves just a small 
bit of extra Java syntax and is easy to understand. The notation <Item> after the class 
name in each of our APIs defines the name Item as a type parameter, a symbolic place¬ 
holder for some concrete type to be used by the client. You can read Stack<Item> as 
“stack of items.” When implementing Stack, we do not know the concrete type of Item, 
but a client can use our stack for any type of data, including one defined long after we 
develop our implementation. The client code provides a concrete type when the stack 
is created: we can replace Item with the name of any reference data type (consistently, 
everywhere it appears). This provides exactly the capability that we need. For example, 
you can write code such as 

Stack<String> stack = new Stack<String>(); 

stack.push("Test"); 

String next = stack. pop(); 
to use a stack for Stri ng objects and code such as 

Queue<Date> queue = new Queue<Date>(); 

queue.enqueue(new Date(12, 31, 1999)); 

Date next = queue.dequeueO ; 

to use a queue for Date objects. If you try to add a Date (or data of any other type than 
Stri ng) to stack or a Stri ng (or data of any other type than Date) to queue, you will 
get a compile-time error. Without generics, we would have to define (and implement) 
different APIs for each type of data we might need to collect; with generics, we can use 
one API (and one implementation) for all types of data, even types that are imple¬ 
mented in the future. As you will soon see, generic types lead to clear client code that is 
easy to understand and debug, so we use them throughout this book. 

Autoboxing. Type parameters have to be instantiated as reference types, so Java has 
special mechanisms to allow generic code to be used with primitive types. Recall that 
Java’s wrapper types are reference types that correspond to primitive types: Boolean, 
Byte, Character, Double, Float, Integer, Long, and Short correspond to boolean, 
byte, char, doubl e, f 1 oat, i nt, 1 ong, and short, respectively. Java automatically con¬ 
verts between these reference types and the corresponding primitive types—in assign¬ 
ments, method arguments, and arithmetic/logic expressions. In the present context, 
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this conversion is helpful because it enables us to use generics with primitive types, as 
in the following code: 

Stack<Integer> stack = new Stack<Integer>() ; 

stack.push(17); // auto-boxing (int -> Integer) 

int i = stack.pop(); // auto-unboxing (Integer -> int) 

Automatically casting a primitive type to a wrapper type is known as autoboxing, and 
automatically casting a wrapper type to a primitive type is known as auto-unboxing. 
In this example, Java automatically casts (autoboxes) the primitive value 17 to be of 
type Integer when we pass it to the push() method. The pop() method returns an 
Integer, which Java casts (auto-unboxes) to an i nt before assigning it to the variable i . 

Iterable collections. For many applications, the client’s requirement is just to process 
each of the items in some way, or to iterate through the items in the collection. This 
paradigm is so important that it has achieved first-class status in Java and many other 
modern languages (the programming language itself has specific mechanisms to sup¬ 
port it, not just the libraries). With it, we can write clear and compact code that is free 
from dependence on the details of a collection’s implementation. For example, suppose 
that a client maintains a collection of transactions in a Queue, as follows: 

Queue<Transaction> collection = new Queue<Transaction>(); 

If the collection is iterable, the client can print a transaction list with a single statement: 

for (Transaction t : collection) 

{ StdOut.println(t); } 

This construct is known as the foreach statement: you can read the f o r statement as for 
each transaction t in the collection, execute the following block of code. This client code 
does not need to know anything about the representation or the implementation of the 
collection; it just wants to process each of the items in the collection. The same f o r loop 
would work with a Bag of transactions or any other iterable collection. We could hardly 
imagine client code that is more clear and compact. As you will see, supporting this 
capability requires extra effort in the implementation, but this effort is well worthwhile. 

it is interesting to note that the only differences between the APIs for Stack and 
Queue are their names and the names of the methods. This observation highlights the 
idea that we cannot easily specify all of the characteristics of a data type in a list of 
method signatures. In this case, the true specification has to do with the English-lan¬ 
guage descriptions that specify the rules by which an item is chosen to be removed (or 
to be processed next in the foreach statement). Differences in these rules are profound, 
part of the API, and certainly of critical importance in developing client code. 
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a bag of 
marbles 



Bags. A bag is a collection where removing items is not supported—its purpose is to 
provide clients with the ability to collect items and then to iterate through the collected 
items (the client can also test if a bag is empty and find its number of items). The order 
of iteration is unspecified and should be immaterial to the client. To appreciate the con¬ 
cept, consider the idea of an avid marble collector, who might put marbles in a bag, one 
at a time, and periodically process all the marbles to look 
for one having some particular characteristic. With our 
Bag API, a client can add items to a bag and process them 
all with a foreach statement whenever needed. Such a cli¬ 
ent could use a stack or a queue, but one way to emphasize 
that the order in which items are processed is immaterial 
is to use a Bag. The class Stats at right illustrates a typi¬ 
cal Bag client. The task is simply to compute the average 
and the sample standard deviation of the double values 
on standard input. If there are N numbers on standard in¬ 
put, their average is computed by adding the numbers and 
dividing by N; their sample standard deviation is comput¬ 
ed by adding the squares of the difference between each 
number and the average, dividing by N— 1, and taking the 
square root. The order in which the numbers are consid¬ 
ered is not relevant for either of these calculations, so we 
save them in a Bag and use the foreach construct to com¬ 
pute each sum. Note : It is possible to compute the standard 
deviation without saving all the numbers (as we did for the 
average in Accumulator —see exercise 1 . 2 . 18 ). Keeping 
the all numbers in a Bag is required for more complicated 
statistics. 




for (Marble m : bag) 


process each marble m 
(in any order) 


Operations on a bag 


www.it-ebooks.info 


1.3 ■ Bags, Queues, and Stacks 


125 


typical Bag client 


application 


public class Stats 

{ 

public static void main(String[] args) 

{ 

Bag<Double> numbers = new Bag<Double>(); 

while ( !Stdln . isEmptyO) 

numbers.add(StdIn. readDoubleO) ; 
int N = numbers.size(); 

double sum = 0.0; 
for (double x : numbers) 
sum += x; 

double mean = sum/N; 
sum = 0.0; 

for (double x : numbers) 

sum += (x - mean)*(x - mean); 
double std = Math.sqrt(sum/(N-1)); 

StdOut.printf("Mean: %.2f\n", mean); 
StdOut.printf("Std dev: %.2f\n", std); 


} 


} 


% java Stats 
100 
99 
101 
120 
98 
107 
109 
81 
101 
90 

Mean: 100.60 
Std dev: 10.51 
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enqueue 

m 


enqueue 


dequeue 


dequeue 


FIFO queues. A FIFO queue (or just a queue) is a collection that is based on the first- 
in-first-out (FIFO) policy. The policy of doing tasks in the same order that they arrive 

is one that we encounter frequently in everyday life: 
queue of customers from people waiting in line at a theater, to cars wait- 

I—| ^ ing in line at a toll booth, to tasks waiting to be ser- 

| o 1 ( i 1 ( 2 ] viced by an application on your computer. One bed¬ 

rock principle of any service policy is the perception 

new arrival . 

at the end of fairness. The first idea that comes to mmd when 

□□ i most people think about fairness is that whoever 

| L°J LiJ L?J CD has been waiting the longest should be served first. 

new arrival That is precisely the FIFO discipline. Queues are a 

at the end natural model for many everyday phenomena, and 

'”TGC HH [— 2 —) pf~) yr~] they play a central role in numerous applications. 

When a client iterates through the items in a queue 
with the foreach construct, the items are processed 
in the order they were added to the queue. A typi¬ 
cal reason to use a queue in an application is to save 
items in a collection while at the same time preserv¬ 
ing their relative order: they come out in the same 
order in which they were put in. For example, the 
client below is a possible implementation of the 
readDoubl es() static method from our In class. 


first in line 
leaves queue 

I □ 

m | mmcura 

next in line 
leaves queue 

I n 


Atypical FIFO queue The problem that this method solves for the client 

is that the client can get numbers from a file into an 
array without knowing the file size ahead of time. We enqueue the numbers from the file, 
use the si ze() method from Queue to find the size needed for the array, create the ar¬ 
ray, and then dequeue the num¬ 
bers to move them to the array. 

A queue is appropriate because 
it puts the numbers into the ar¬ 
ray in the order in which they 
appear in the file (we might use 
a Bag if that order is immateri¬ 
al). This code uses autoboxing 
and auto-unboxing to convert 
between the client’s double 
primitive type and and the 
queue’s Double wrapper type. 


public static int[] readInts(String name) 

{ 

In in = new In(name); 

Queue<Integer> q = new Queue<Integer>(); 
while (Iin.isEmpty()) 

q. enqueue(i n . readlntO) ; 

i nt N = q. size() ; 
i nt [] a = new i nt [N] ; 
for (int i =0; i < N; i++) 
a[i ] = q . dequeueO ; 
return a; 


Sample Queue client 


www.it-ebooks.info 


1.3 ■ Bags, Queues, and Stacks 


127 


Pushdown stacks. A pushdown stack (or just a stack) is 
a collection that is based on the last-in-first-out (LIFO) 
policy. When you keep your mail in a pile on your desk, 
you are using a stack. You pile pieces of new mail on the 
top when they arrive and take each piece of mail from 
the top when you are ready to read it. People do not 
process as many papers as they did in the past, but the 
same organizing principle underlies several of the ap¬ 
plications that you use regularly on your computer. For 
example, many people organize their email as a stack— 
they push messages on the top when they are received 
and pop them from the top when they read them, with 
most recently received first (last in, first out). The ad¬ 
vantage of this strategy is that we see interesting email as 
soon as possible; the disadvantage is that some old email 
might never get read if we never empty the stack. You 
have likely encountered another common example of a 
stack when surfing the web. When you click a hyperlink, 
your browser displays the new page (and pushes onto a 
stack). You can keep clicking on hyperlinks to visit new 
pages, but you can always revisit the previous page by 
clicking the back button (popping it from the stack). 
The LIFO policy offered by a stack provides just the be¬ 
havior that you expect. When a client iterates through 
the items in a stack with the foreach construct, the items 
are processed in the reverse of the order in 
which they were added. A typical reason to 
use a stack iterator in an application is to save 
items in a collection while at the same time 
reversing their relative order. For example, 
the client Reverse at right reverses the or¬ 
der of the integers on standard input, again 
without having to know ahead of time how 
many there are. The importance of stacks in 
computing is fundamental and profound, 
as indicated in the detailed example that we 
consider next. 


a stack of 
documents 


push ( 


push 




'= pop() 



r = pop() 


ZI 


new (gray) one 
goes on top 


new (black) one 
goes on top 


remove the 
black one 
from the top 


remove the 
- gray one 
from the top 


Operations on a pushdown stack 


public 
{ 


class Reverse 


public static void mai n(String[] args) 
{ 

Stack<Integer> stack; 
stack = new Stack<Integer>(); 
while (! Stdln.i sEmptyO) 

stack, push(Stdln . readlntQ) ; 


for (int i : stack) 
StdOut.println(i) ; 


} 


} 


Sample Stack client 
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Arithmetic expression evaluation. As another example of a stack client, we consider 
a classic example that also demonstrates the utility of generics. Some of the first pro¬ 
grams that we considered in section 1.1 involved computing the value of arithmetic 
expressions like this one: 

(l+((2 + 3)*(4*5))) 

If you multiply 4 by 5, add 3 to 2, multiply the result, and then add 1, you get the value 
101 . But how does the Java system do this calculation? Without going into the details of 
how the Java system is built, we can address the essential ideas by writing a Java program 
that can take a string as input (the expression) and produce the number represented by 
the expression as output. For simplicity, we begin with the following explicit recursive 
definition: an arithmetic expression is either a number, or a left parenthesis followed by 
an arithmetic expression followed by an operator followed by another arithmetic ex¬ 
pression followed by a right parenthesis. For simplicity, this definition is ion fully paren¬ 
thesized arithmetic expressions, which specify precisely which operators apply to which 
operands—you are a bit more familiar with expressions such as 1 + 2 * 3, where we 
often rely on precedence rules instead of parentheses. The same basic mechanisms that 
we consider can handle precedence rules, but we avoid that complication. For speci¬ 
ficity, we support the familiar binary operators *, +, -, and /, as well as a square-root 
operator sqrt that takes just one argument. We could easily allow more operators and 
more kinds of operators to embrace a large class of familiar mathematical expressions, 
involving trigonometric, exponential, and logarithmic functions. Our focus is on un¬ 
derstanding how to interpret the string of parentheses, operators, and numbers to en¬ 
able performing in the proper order the low-level arithmetic operations that are avail¬ 
able on any computer. Precisely how can we convert an arithmetic expression—a string 
of characters—to the value that it represents? A remarkably simple algorithm that was 
developed by E. W. Dijkstra in the 1960s uses two stacks (one for operands and one for 
operators) to do this job. An expression consists of parentheses, operators, and oper¬ 
ands (numbers). Proceeding from left to right and taking these entities one at a time, 
we manipulate the stacks according to four possible cases, as follows: 

■ Push operands onto the operand stack. 

■ Push operators onto the operator stack. 

■ Ignore left parentheses. 

■ On encountering a right parenthesis, pop an operator, pop the requisite number 
of operands, and push onto the operand stack the result of applying that opera¬ 
tor to those operands. 

After the final right parenthesis has been processed, there is one value on the stack, 
which is the value of the expression. This method may seem mysterious at first, but it 
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Dijkstra's Two-Stack Algorithm for Expression Evaluation 


pub!ic 

{ 


class Evaluate 


public static void main(String[] args) 

{ 

Stack<String> ops = new Stack<String>(); 
Stack<Double> vals = new Stack<Doubl e>() ; 
while (!Stdln.isEmptyO) 

{ // Read token, push if operator. 

String s = Stdln.readString(); 


if 

else if 
else if 
else if 
else if 
else if 
else if 
{ // Pop, evaluate, and 

String op = ops.popO 
double v = vals.popQ 


(s.equals("(")) 
(s.equals("+")) 
Cs.equalsC"-")) 
Cs.equalsC"*")) 
Cs.equalsC"/")) 

Cs. equal sC'sqrt")) 
Cs.equalsC")")) 

push 


ops.pushCs) 
ops.pushCs) 
ops.pushCs) 
ops.pushCs) 
ops.pushCs) 


result if token is ")". 


if 

el se 
el se 
el se 
el se 


vals.popC) 
vals.popC) 
vals.popC) 
vals.popC) 


/ 


Cop.equalsC"+")) v 

if Cop.equalsC"-")) v 

if Cop.equalsC"*")) v 

if Cop.equalsC"/")) v 

if Cop.equalsC'sqrt")) v 
vals.pushCv); 

} // Token not operator or paren: push double value, 

else vals.pushCDouble.parseDoubleCs)) ; 

} 

StdOut.printlnCvals.popO); 


Math.sqrtCv); 


This Stack client uses two stacks to evaluate arithmetic expressions, illustrating an essential compu¬ 
tational process: interpreting a string as a program and executing that program to compute the de¬ 
sired result. With generics, we can use the code in a single Stack implementation to implement one 


stack of String values and another stack of Double 


values. For simplicity, this code assumes that the expres¬ 
sion is fully parenthesized, with numbers and characters 

% java Evaluate 

C1+CC2 + 3)*C4*5))) 

separated by whitespace. 

101.0 


% java Evaluate 

C C 1 + sqrt C 5.0 ) ) / 2.0 ) 
1.618033988749895 


1 
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is easy to convince yourself that it computes the proper value: any time the algorithm 
encounters a subexpression consisting of two operands separated by an operator, all 
surrounded by parentheses, it leaves the result of performing that operation on those 
operands on the operand stack. The result is the same as if that value had appeared in 
the input instead of the subexpression, so we can think of replacing the subexpression 
by the value to get an expression that would yield the same result. We can apply this 
argument again and again until we get a single value. For example, the algorithm com¬ 
putes the same value for all of these expressions: 

(l+((2 + 3)*(4*5))) 

(l+(5*(4*5))) 

( 1 + ( 5 * 20 ) ) 

( 1 + 100 ) 

101 

Eval uate on the previous page is an implementation of this algorithm. This code is a 
simple example of an interpreter: a program that interprets the computation specified 
by a given string and performs the computation to arrive at the result. 
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operand 
stack v. 


operator 
stack n. 





left parenthesis: ignore 


(1+C(2+B)*(4*5))) 
operand: push onto operand stack 
1 + ( ( 2 + B ) * ( 4 * 5 ) ) ) 
^ operator: push onto operator stack 
+ ( ( 2 + 3 ) * ( 4 * 5 ) ) ) 


((2 + 3)*(4*5))) 


(2+3)*(4*5))) 


2 + 3) *(4*5))) 


+ 3 ) * ( 4 * 5 ) ) ) 


3 ) * ( 4 * 5 ) ) ) 

right parenthesis: pop operator 
^ and operands ana push result 

) * ( 4 * 5 ) ) ) 


*(4*5))) 


(4*5))) 


4*5))) 
* 5 ) ) ) 

5 ) ) ) 

) ) ) 

) ) 

) 


Trace of Dijkstra's two-stack arithmetic expression-evaluation algorithm 
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CHAPTER 1 


Fundamentals 


Implementing collections To address the issue of implementing Bag, Stack and 
Queue, we begin with a simple classic implementation, then address improvements that 
lead us to implementations of the APIs articulated on page 121. 

Fixed-capacity stack. As a strawman, we consider an abstract data type for a fixed- 
capacity stack of strings, shown on the opposite page. The API differs from our Stack 
API: it works only for Stri ng values, it requires the client to specify a capacity, and it 
does not support iteration. The primary choice in developing an API implementation is 
to choose a representation for the data. For Fi xedCapaci tyStackOf St ri ngs, an obvious 
choice is to use an array of Stri ng values. Pursuing this choice leads to the implemen¬ 
tation shown at the bottom on the opposite page, which could hardly be simpler (each 
method is a one-liner). The instance variables are an array a[] that holds the items in 
the stack and an integer N that counts the number of items in the stack. To remove an 
item, we decrement N and then return a [N]; to insert a new item, we set a [N] equal to 
the new item and then increment N. These operations preserve the following properties: 

■ The items in the array are in their insertion order. 

■ The stack is empty when N is 0. 

■ The top of the stack (if it is nonempty) is at a [N-l]. 

As usual, thinking in terms of invariants of this sort is the easiest way to verify that an 
implementation operates as intended. Be sure that you fully understand this implemen¬ 
tation. The best way to do so is to examine a trace of the stack contents for a sequence of 

operations, as illustrated at left for the test client, 
which reads strings from standard input and push¬ 
es each string onto a stack, unless it is when it 
pops the stack and prints the result. The primary 
performance characteristic of this implementation 
is that the push and pop operations take time inde¬ 
pendent of the stack size. For many applications, it 
is the method of choice because of its simplicity. 
But it has several drawbacks that limit its potential 
applicability as a general-purpose tool, which we 
now address. With a moderate amount of effort 
(and some help from Java language mechanisms), 
we can develop an implementation that is broadly 
useful. This effort is worthwhile because the im¬ 
plementations that we develop serve as a model for 
implementations of other, more powerful, abstract 
Trace of FixedCapacityStackOfStrings test client data tyP es throughout the book. 
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API public class FixedCapacityStackOfStrings 



FixedCapacityStackOfStrings(int cap) 

create an empty stack of capacity cap 

voi d 

push(String item) 

add a string 

Stri ng 

pop() 

remove the most recently added string 

boolean 

i sEmptyO 

is the stack empty? 

i nt 

size() 

number of strings on the stack 


testclient public static void main(Stri ng[] args) 

{ 

FixedCapacityStackOfStrings s; 
s = new FixedCapacityStackOfStrings(100); 
while ( !Stdln . isEmptyO) 

{ 

String item = Stdln . readStri ng() ; 
if (litem.equals("-")) 
s.push(item) ; 

else if (! s . isEmptyO) StdOut. print(s. pop() + " "); 

} 

StdOut.println("(" + s.size() + " left on stack)"); 

} 


application % m0 re tobe.txt 

to be or not to-be - - that - - - is 

% java FixedCapacityStackOfStrings < tobe.txt 
to be not that or be (2 left on stack) 


iplementation public class FixedCapacityStackOfStrings 

{ 

private String[] a; // stack entries 
private int N; // size 

public FixedCapacityStackOfStrings (int cap) 

{ a = new String [cap]; } 

public boolean isEmptyO { return N == 0; } 
public int size() { return N; } 

public void push (String item) 

{ a[N++] = item; } 

public String pop() 

{ return a[—N] ; } 


} 

An abstract data type for a fixed-capacity stack of strings 
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Generics. The first drawback of Fi xedCapaci tyStackOfStri ngs is that it works only 
for String objects. If we want a stack of double values, we would need to develop 
another class with similar code, essentially replacing Stri ng with doubl e everywhere. 
This is easy enough but becomes burdensome when we consider building a stack of 
Transacti on values or a queue of Date values, and so forth. As discussed on page 122, 
Java’s parameterized types (generics) are specifically designed to address this situation, 
and we saw several examples of client code (on pages 125,126,127, and 129). But how 
do we implement a generic stack? The code on the facing page shows the details. It imple¬ 
ments a class Fi xedCapaci tyStack that differs from Fi xedCapaci tyStackOfStri ngs 
only in the code highlighted in red—we replace every occurrence of Stri ng with Item 
(with one exception, discussed below) and declare the class with the following first line 
of code: 

public class FixedCapacityStack<Item> 

The name Item is a type parameter, a symbolic placeholder for some concrete type to be 
used by the client. You can read Fi xedCapaci tyStack<Item> as stack of items, which is 
precisely what we want. When implementing Fi xedCapaci tyStack, we do not know 
the actual type of Item, but a client can use our stack for any type of data by providing a 
concrete type when the stack is created. Concrete types must be reference types, but cli¬ 
ents can depend on autoboxing to convert primitive types to their corresponding wrap¬ 
per types. Java uses the type parameter Item to check for type mismatch errors—even 
though no concrete type is yet known, variables of type Item must be assigned values 
of type Item, and so forth. But there is one significant hitch in this story: We would like 
to implement the constructor in Fi xedCapaci tyStack with the code 

a = new Itemjcap]; 

which calls for creation of a generic array. For historical and technical reasons beyond 
our scope, generic array creation is disallowed in Java. Instead, we need to use a cast: 

a = (ItemJJ) new Objectjcap]; 

This code produces the desired effect (though the Java compiler gives a warning, which 
we can safely ignore), and we use this idiom throughout the book (the Java system li¬ 
brary implementations of similar abstract data types use the same idiom). 
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API 


test client 


public class FixedCapacityStack<Item> 



FixedCapacityStack(int cap) 

create an empty stack of capacity cap 

voi d 

push (Item item) 

add an item 

Item 

pop() 

remove the most recently added item 

boolean 

i sEmptyO 

is the stack empty? 

int 

size() 

number of items on the stack 

public 

static void main(String[] args) 



{ 

FixedCapacityStack<String> s; 
s = new FixedCapacityStack<String>(100); 
while (! Stdln. i sEmptyO) 

{ 

String item = Stdln . readStri ng() ; 
if (litem.equals("-")) 
s . push(i tern) ; 

else if (! s. i sEmptyO) StdOut.print(s.pop() + " "); 

} 


StdOut.println("(" + s.sizeO + " left on stack)"); 

} 


application % more tobe.txt 

to be or not to-be - - that - - - is 

% java FixedCapacityStack < tobe.txt 
to be not that or be (2 left on stack) 


iplementation public class FixedCapaci tyStack<Item> 

{ 

private Item[] a; // stack entries 
private int N; // size 

public Fi xedCapaci tyStack(int cap) 

{ a = (Item[]) new Object[cap]; } 

public boolean isEmptyO { return N == 0; } 
public int size() { return N; } 

public void push(Item item) 

{ a[N++] = item; } 

public Item pop() 

{ return a[--N]; } 


An abstract data type for a fixed-capacity generic stack 
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Array resizing. Choosing an array to represent the stack contents implies that clients 
must estimate the maximum size of the stack ahead of time. In Java, we cannot change 
the size of an array once created, so the stack always uses space proportional to that 
maximum. A client that chooses a large capacity risks wasting a large amount of mem¬ 
ory at times when the collection is empty or nearly empty. For example, a transaction 
system might involve billions of items and thousands of collections of them. Such a 
client would have to allow for the possibility that each of those collections could hold 
all of those items, even though a typical constraint in such systems is that each item 
can appear in only one collection. Moreover, every client risks overflow if the collection 
grows larger than the array. For this reason, push() needs code to test for a full stack 
,and we should have an i sFul 1 () method in the API to allow clients to test for that 
condition. We omit that code, because our desire is to relieve the client from having to 
deal with the concept of a full stack, as articulated in our original Stack API. Instead, 
we modify the array implementation to dynamically adjust the size of the array a [] so 
that it is both sufficiently large to hold all of the items and not so large as to waste an 
excessive amount of space. Achieving these goals turns out to be remarkably easy. First, 
we implement a method that moves a stack into an array of a different size: 

private void resize(int max) 

{ // Move stack of size N <= max to a new array of size max. 

Item[] temp = (Item[]) new Object[max]; 
for (int i =0; i < N; i++) 
temp[i] = a[i] ; 
a = temp; 

} 

Now, in push O, we check whether the array is too small. In particular, we check wheth¬ 
er there is room for the new item in the array by checking whether the stack size N is 
equal to the array size a. 1 ength. If there is no room, we double the size of the array. 
Then we simply insert the new item with the code a [N++] = item, as before: 

public void push(String item) 

{ // Add item to top of stack. 

if (N == a.length) resize(2*a.length); 
a[N++] = item; 

} 

Similarly, in pop(), we begin by deleting the item, then we halve the array size if it is 
too large. If you think a bit about the situation, you will see that the appropriate test 
is whether the stack size is less than one-fourth the array size. After the array is halved, 
it will be about half full and can accommodate a substantial number of push() and 
pop() operations before having to change the size of the array again. 
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public String pop() 

{ // Remove item from top of stack. 

String item = a[ —N]; 

a[N] = null; // Avoid loitering (see text). 

if (N > 0 && N == a.length/4) resize(a.length/2); 

return item; 

} 

With this implementation, the stack never overflows and never becomes less than one- 
quarter full (unless the stack is empty, when the array size is 1). We will address the 
performance analysis of this approach in more detail in section 1 . 4 . 

Loitering. Java’s garbage collection policy is to reclaim the memory associated with 
any objects that can no longer be accessed. In our pop() implementations, the refer¬ 
ence to the popped item remains in the array. The item is effectively an orphan —it will 
be never be accessed again—but the Java garbage collector has no way to know this 
until it is overwritten. Even when the client is done with the item, the reference in the 
array may keep it alive. This condition (holding a reference to an item that is no longer 
needed) is known as loitering. In this case, loitering is easy to avoid, by setting the array 
entry corresponding to the popped item to null, thus overwriting the unused refer¬ 
ence and making it possible for the system to reclaim the memory associated with the 
popped item when the client is finished with it. 


pushQ pop() N a.length 


a[] 

3 4 5 6 7 



Trace of array resizing during a sequence of pushO and pop() operations 


wwwJt-ebooks.info 









138 


CHAPTER 1 ■ Fundamentals 


Iteration. As mentioned earlier in this section, one of the fundamental operations on 
collections is to process each item by iterating through the collection using Java’s foreach 
statement. This paradigm leads to clear and compact code that is free from dependence 
on the details of a collection’s implementation. To consider the task of implementing 
iteration, we start with a snippet of client code that prints all of the items in a collection 
of strings, one per line: 

Stack<String> collection = new Stack<String>(); 

for (String s : collection) 

StdOut.println(s); 

Now, this foreach statement is shorthand for a whi 1 e construct (just like the for state¬ 
ment itself). It is essentially equivalent to the following while statement: 

Iterator<String> i = collection.iteratorO; 
while (i .hasNextO) 

{ 

String s = i.next(); 

StdOut.println(s); 

} 

This code exposes the ingredients that we need to implement in any iterable collection: 

■ The collection must implement an i terator () method that returns an 
Iterator object. 

■ The Iterator class must include two methods: hasNextO (which returns a 
boolean value) and next() (which returns a generic item from the collection). 

In Java, we use the i nterface mechanism to express the idea that a class implements 
a specific method (see page 100). For iterable collections, the necessary interfaces are al¬ 
ready defined for us in Java. To make a class iterable, the first step is to add the phrase 
i mpl ements Iterabl e<Item> to its declaration, matching the interface 

public interface Iterable<Item> 

{ 

Iterator<Item> iteratorO; 

} 

(which is in java. 1 ang. Iterabl e), and to add a method i teratorO to the class that 
returns an Iterator<Item>. Iterators are generic, so we can use our parameterized 
type Item to allow clients to iterate through objects of whatever type is provided by our 
client. For the array representation that we have been using, we need to iterate through 
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an array in reverse order, so we name the iterator ReverseAr ray Iterator and add this 
method: 

public Iterator<Item> iteratorO 
{ return new ReverseArraylterator(); } 

What is an iterator? An object from a class that implements the methods hasNextO 
and next(), as defined in the following interface (which is in java, uti 1. Iterator): 

public interface Iterator<Item> 

{ 

boolean hasNextO; 

Item next(); 
void remove(); 

} 

Although the interface specifies a remove () method, we always use an empty method 
for remove () in this book, because interleaving iteration with operations that modify 
the data structure is best avoided. For ReverseArraylterator, these methods are all 
one-liners, implemented in a nested class within our stack class: 

private class ReverseArraylterator implements Iterator<Item> 

{ 

private int i = N; 

public boolean hasNextO 
public Item next() 
public void removeO 

} 

Note that this nested class can access the instance variables of the enclosing class, in 
this case a[] and N (this ability is the main reason we use nested classes for iterators). 
Technically, to conform to the Iterator specification, we should throw exceptions 
in two cases: an UnsupportedOperationException if a client calls removeO and a 
NoSuchEl ementExcepti on if a client calls nextO when i is 0. Since we only use itera¬ 
tors in theforeach construction where these conditions do not arise, we omit this code. 
One crucial detail remains: we have to include 

import java.util.Iterator; 

at the beginning of the program because (for historical reasons) Iterator is not part 
of java.lang (even though Iterable is part of java.lang). Now a client using the 
foreach statement for this class will get behavior equivalent to the common f o r loop for 
arrays, but does not need to be aware of the array representation (an implementation 


{ return i > 0; } 

{ return a[--i]; } 

{ } 
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detail). This arrangement is of critical importance for implementations of fundamen¬ 
tal data types like the collections that we consider in this book and those included in 
)ava libraries. For example, it frees us to switch to a totally different representation 
without having to change any client code. More important, taking the client’s point of 
view, it allows clients to use iteration without having to know any details of the class 
implementation. 

algorithm l.i is an implementation of our Stack API that resizes the array, allows 
clients to make stacks for any type of data, and supports client use of foreach to iterate 
through the stack items in LIFO order. This implementation is based on lava language 
nuances involving Iterator and Iterabl e,but there is no need to study those nuances 
in detail, as the code itself is not complicated and can be used as a template for other 
collection implementations. 

For example, we can implement the Queue API by maintaining two indices as in¬ 
stance variables, a variable head for the beginning of the queue and a variable tai 1 for 
the end of the queue. To remove an item, use head to access it and then increment head; 
to insert an item, use tai 1 to store it, and then increment tai 1. If incrementing an 
index brings it past the end of the array, reset it to 0. Developing the details of checking 
when the queue is empty and when the array is full and needs resizing is an interesting 
and worthwhile programming exercise (see exercise 1 . 3 . 14 ). 
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Trace of ResizingArrayQueue test client 


In the context of the study of algorithms, algorithm 1.1 is significant because 
it almost (but not quite) achieves optimum performance goals for any collection 
implementation: 

■ Each operation should require time independent of the collection size. 

■ The space used should always be within a constant factor of the collection size. 
The flaw in Resi zi ngArrayStack is that some push and pop operations require resiz¬ 
ing: this takes time proportional to the size of the stack. Next, we consider a way to cor¬ 
rect this flaw, using a fundamentally different way to structure data. 
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ALGORITHM 1.1 Pushdown (LIFO) stack (resizing array implementation) 


import java.util.Iterator; 

public class ResizingArrayStack<Item> implements Iterable<Item> 

{ 


private Item[] a = (Item[]) new Object[l]; 
private int N = 0; 

public boolean isEmptyO { return N == 0; 
public int sizeQ { return N; 


// stack items 
// number of items 

} 

} 


private void resize(int max) 

{ // Move stack to a new array of size max. 

Item[] temp = (Item[]) new Object[max]; 
for (int i =0; i < N; i++) 
temp[i] = a[i]; 
a = temp; 


} 


public void push(Item 

{ // Add item to top 

if (N == a.length) 
a[N++] = item; 

} 


i tern) 
of stack. 

resize(2*a.length); 


public Item pop() 

{ // Remove item from top of stack. 

Item itern = a[--N]; 

a[N] = null; // Avoid loitering (see text). 

if (N > 0 && N == a.length/4) resize(a.length/2); 
return item; 


public Iterator<Item> iterator() 

{ return new ReverseArraylteratorQ; } 


} 


private class ReverseArraylte 
{ // Support LIFO iteration. 

private int i = N; 
public boolean hasNextO { 
public Item next() { 
public void removeQ { 


rator implements 


return i > 0; 
return a[--i]; 


} 


Iterator<Item> 


} 

} 

} 


This generic, iterable implementation of our Stack API is a model for collection ADTs that keep 
items in an array. It resizes the array to keep the array size within a constant factor of the stack size. 
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Linked lists Now we consider the use of a fundamental data structure that is an ap¬ 
propriate choice for representing the data in a collection ADT implementation. This is 
our first example of building a data structure that is not directly supported by the Java 
language. Our implementation serves as a model for the code that we use for building 
more complex data structures throughout the book, so you should read this section 
carefully, even if you have experience working with linked lists. 

Definition. A linked list is a recursive data structure that is either empty (null) or a 
reference to a node having a generic item and a reference to a linked list. 


The node in this definition is an abstract entity that might hold any kind of data, in ad¬ 
dition to the node reference that characterizes its role in building linked lists. As with a 
recursive program, the concept of a recursive data structure can be a bit mindbending 
at first, but is of great value because of its simplicity. 

Node record. With object-oriented programming, implementing linked lists is not dif¬ 
ficult. We start with a nested class that defines the node abstraction: 

private class Node 

{ 

Item item; 

Node next; 

} 

A Node has two instance variables: an Item (a parameterized type) and a Node. We 
define Node within the class where we want to use it, and make it private because it 
is not for use by clients. As with any data type, we create an object of type Node by in¬ 
voking the (no-argument) constructor with new Node(). The result is a reference to a 
Node object whose instance variables are both initialized to the value null. The Item is 
a placeholder for any data that we might want to structure with a linked list (we will use 
Java’s generic mechanism so that it can represent any reference type); the instance vari¬ 
able of type Node characterizes the linked nature of the data structure. To emphasize 
that we are just using the Node class to structure the data, we define no methods and 
we refer directly to the instance variables in code: if f i rst is a variable associated with 
an object of type Node, we can refer to the instance variables with the code fi rst. i tern 
and f i rst. next. Classes of this kind are sometimes called records. They do not imple¬ 
ment abstract data types because we refer directly to instance variables. However, Node 
and its client code are in the same class in all of our implementations and not accessible 
by clients of that class, so we still enjoy the benefits of data abstraction. 
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Building a linked list. Now, from the recursive definition, we can represent a linked 
list with a variable of type Node simply by ensuring that its value is either null or a ref¬ 
erence to a Node whose next field is a reference to a linked list. For example, to build a 
linked list that contains the items to, be, and or, we create a Node for each item: 


Node first = new Node(); 

Node second = new Node(); 

Node third = new Node(); 

and set the item field in each of the nodes to the 
desired value (for simplicity, these examples assume 
that Item is Stri ng): 

first.item = "to"; 
second.itern = "be"; 
third.item = "or"; 

and set the next fields to build the linked list: 


Node first = new Node(); 
first, item = "to"; 


fi rst 



Node second = new Node(); 
second.item = "be"; 
first.next = second; 


first.next = second; 
second.next = third; 

(Note that thi rd. next remains nul 1 , the value it 
was initialized to at the time of creation.) As a result, 
thi rd is a linked list (it is a reference to a node that 
has a reference to null, which is the null reference 
to an empty linked list), and second is a linked list 
(it is a reference to a node that has a reference to 
thi rd, which is a linked list), and fi rst is a linked 
list (it is a reference to a node that has a reference to 
second, which is a linked list). The code that we will 
examine does these assignment statements in a dif¬ 
ferent order, depicted in the diagram on this page. 


fi rst 


second 



Node third = new Node(); 
third.item = "or"; 
second.next = third; 



Linking together a list 


a linked list represents a sequence of items. In the example just considered, f i r st 
represents the sequence to be or. We can also use an array to represent a sequence of 
items. For example, we cotild use 

String[] s = { "to", "be", "or" }; 

to represent the same sequence of strings. The difference is that it is easier to insert 
items into the sequence and to remove items from the sequence with linked lists. Next, 
we consider code to accomplish these tasks. 
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When tracing code that uses linked lists and other linked structures, we use a visual 
representation where 

■ We draw a rectangle to represent each object 

■ We put the values of instance variables within the rectangle 

■ We use arrows that point to the referenced objects to depict references 

This visual representation captures the essential characteristic of linked lists. For econ¬ 
omy, we use the term links to refer to node references. For simplicity, when item values 
are strings (as in our examples), we put the string within the object rectangle rather 
than the more accurate rendition depicting the string object and the character array 
that we discussed in section 1 . 2 . This visual representation allows us to focus on the 
links. 

Insert at the beginning. First, suppose that you want to insert a new node into a linked 
list. The easiest place to do so is at the beginning of the list. For example, to insert the 
string not at the beginning of a given linked list whose first node is fi rst, we save 
fi rst in ol dfi rst, assign to fi rst a new Node, and assign its i tem field to not and its 
next field to ol df i rst. This code for inserting a node at the beginning of a linked list 
involves just a few assignment statements, so the amount of time that it takes is inde¬ 
pendent of the length of the list. 

save a link to the list 

Node oldfirst = first; 


oldfirst 
\ 



create a new node for the beginning 
fi rst = new NodeQ ; 


oldfirst 



set the instance variables in the new node 

first.item = "not"; 
first, next = oldfirst; 



Inserting a new node at the beginning of a linked list 
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Remove from the beginning. Next, suppose that you first = first.next; 
want to remove the first node from a list. This op¬ 
eration is even easier: simply assign to fi rst the value 
fi rst. next. Normally, you would retrieve the value of 
the item (by assigning it to some variable of type Item) 
before doing this assignment, because once you change 
the value of fi rst, you may not have any access to the 
node to which it was referring. Typically, the node ob¬ 
ject becomes an orphan, and the Java memory manage¬ 
ment system eventually reclaims the memory it occupies. 

Again, this operation just involves one assignment statement, so its running time is 
independent of the length of the list. 



1 ° r 


| null | 





1 ° r 


| null | 



Removing the first node in a linked list 


Insert at the end. How do we add a node to the end of a linked list? To do so, we need 
a link to the last node in the list, because that node’s link has to be changed to refer¬ 
ence a new node containing the item to be inserted. Maintaining an extra link is not 
something that should be taken lightly in linked-list code, because every method that 
modifies the list needs code to check whether that variable needs to be modified (and 


to make the necessary modifications). For 
example, the code that we just examined for 
removing the first node in the list might in¬ 
volve changing the reference to the last node 
in the list, since when there is only one node 
in the list, it is both the first one and the last 
one! Also, this code does not work (it follows 
a null link) in the case that the list is empty. 
Details like these make linked-list code noto¬ 
riously difficult to debug. 

Insert/remove at other positions. In sum¬ 
mary, we have shown that we can implement 
the following operations on linked lists with 
just a few instructions, provided that we have 
access to both a link fi rst to the first ele¬ 
ment in the list and a link last to the last 
element in the list: 

■ Insert at the beginning. 

■ Remove from the beginning. 

■ Insert at the end. 


save a link to the last node 

Node oldlast = last; 

oldlast 



create a new node for the end 

Node last = new Node(); 
last.item = "not"; 


oldlast 



link the new node to the end of the list 

oldlast.next = last; 



Inserting a new node at the end of a linked list 
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Other operations, such as the following, are not so easily handled: 

■ Remove a given node. 

■ Insert a new node before a given node. 

For example, how can we remove the last node from a list? The link I ast is no help, 
because we need to set the link in the previous node in the list (the one with the same 
value as 1 ast) to null . In the absence of any other information, the only solution is to 
traverse the entire list looking for the node that links to 1 ast (see below and exercise 
1.3.19). Such a solution is undesirable because it takes time proportional to the length 
of the list. The standard solution to enable arbitrary insertions and deletions is to use 
a doubly-linked list, where each node has two links, one in each direction. We leave the 
code for these operations as an exercise (see exercise 1.3.31). We do not need doubly 
linked lists for any of our implementations. 

Traversal. To examine every item in an array, we use familiar code like the following 
loop for processing the items in an array a []: 

for (int i =0; i < N; i++) 

{ 

// Process a[i]. 

} 

There is a corresponding idiom for examining the items in a linked list: We initialize a 
loop index variable x to reference the first Node of the linked list. Then we find the item 
associated with x by accessing x . i tem, and then update x to refer to the next Node in the 
linked list, assigning to it the value of x. next and repeating this process until x is nul 1 
(which indicates that we have reached the end of the linked list). This process is known 
as traversing the list and is succinctly expressed in code like the following loop for pro¬ 
cessing the items in a linked list whose first item is associated with the variable f i rst: 

for (Node x = first; x != null; x = x.next) 

{ 

// Process x.item. 

} 

This idiom is as natural as the standard idiom for iterating through the items in an ar¬ 
ray. In our implementations, we use it as the basis for iterators for providing client code 
the capability of iterating through the items, without having to know the details of the 
linked-list implementation. 
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Stack implementation. Given these preliminaries, developing an implementation for 
our Stack API is straightforward, as shown in algorithm 1.2 on page 149. It maintains 
the stack as a linked list, with the top of the stack at the beginning, referenced by an 
instance variable fi rst. Thus, to push() an item, we add it to the beginning of the 
list, using the code discussed on page 144 and to pop() an item, we remove it from the 
beginning of the list, using the code discussed on page 145. To implement s i ze (), we keep 
track of the number of items in an instance variable N, incrementing N when we push 
and decrementing N when we pop. To implement i sEmptyO we check whether fi rst 
is null (alternatively, we could check whether N is 0 ). The implementation uses the 
generic type Item —you can think of the code <Item> after the class name as meaning 
that any occurrence of Item in the implementation will be replaced by a client-supplied 
data-type name (see page 134) . For now, we omit the code to support iteration, which we 
consider on page 155. A trace for the test client that we have been using is shown on the 
next page. This use of linked lists achieves our optimum design goals: 

■ It can be used for any type of data. 

■ The space required is always proportional to the size of the collection. 

■ The time per operation is always independent of the size of the collection. 

This implementation is a prototype for many algorithm implementations that we con¬ 
sider. It defines the linked-list data structure and implements the client methods push () 
and popO that achieve the specified effect with just a few lines of code. The algorithms 
and data structure go hand in hand. In this case, the code for the algorithm implemen¬ 
tations is quite simple, but the properties of the data structure are not at all elemen¬ 
tary, requiring explanations on the past several pages. This interaction between data 
structure definition and algorithm implementation is typical and is our focus in ADT 
implementations throughout this book. 

public static void main(String[] args) 

{ // Create a stack and push/pop strings as directed on Stdln. 

Stack<String> s = new Stack<String>() ; 

while (!Stdln . isEmptyO) 

{ 

String item = Stdln. readString() ; 

if (litem.equals("-")) 
s.push(item) ; 

else if (! s . isEmptyO) StdOut.pri nt(s. pop() + " ") ; 

} 

StdOut. pri ntln("(" + s.size() + " left on stack)"); 

} 

Test client for Stack 
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Stdln StdOut 


to 



be 


or 


not 


to 


be 


that 


to 


be 


not 


that 


or 



to 

~nuU 







Trace of Stack development client 
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ALGORITHM 1.2 Pushdown stack (linked-list implementation) 


public class Stack<Item> implements Iterable<Item> 

{ 

private Node first; // top of stack (most recently added node) 
private int N; // number of items 

private class Node 

{ // nested class to define nodes 

Item item; 

Node next; 

} 

public boolean isEmptyO { return first == null; } // Or: N == 0. 

public int size() { return N; } 

public void push(Item item) 

{ // Add item to top of stack. 

Node oldfirst = first; 
first = new Node(); 
first.item = item; 
first.next = oldfirst; 

N++; 

} 

public Item pop() 

{ // Remove item from top of stack. 

Item item = first.item; 
fi rst = first.next; 

N—; 

return item; 

} 

// See page 155 for iteratorO implementation. 

// See page 147 for test client main(). 

} 


This generic Stack implementation is based on a linked-list data structure. It can be used to create 
stacks containing any type of data. To support 
iteration, add the highlighted code described 
for Bag on page 155 . 


% more tobe.txt 

to be or not to-be - - that - - - is 

% java Stack < tobe.txt 

to be not that or be (2 left on stack) 
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Queue implementation. An implementation of our Queue API based on the linked- 
list data structure is also straightforward, as shown in algorithm 1.3 on the facing 
page. It maintains the queue as a linked list in order from least recently to most recently 
added items, with the beginning of the queue referenced by an instance variable f i rst 
and the end of the queue referenced by an instance variable last. Thus, to enqueue 0 
an item, we add it to the end of the list (using the code discussed on page 145 , augmented 
to set both fi rst and last to refer to the new node when the list is empty) and to 
dequeue () an item, we remove it from the beginning of the list (using the same code 
as for popO in Stack, augmented to update last when the list becomes empty). The 
implementations of si ze() and i sEmptyO are the same as for Stack. As with Stack 
the implementation uses the generic type parameter Item, and we omit the code to 
support iteration, which we consider in our Bag implementation on page 155 . A develop¬ 
ment client similar to the one we used for Stack is shown below, and the trace for this 
client is shown on the following page. This implementation uses the same data struc¬ 
ture as does Stack—a linked list—but it implements different algorithms for adding 
and removing items, which make the difference between LIFO and FIFO for the client. 
Again, the use of linked lists achieves our optimum design goals: it can be used for any 
type of data, the space required is proportional to the number of items in the collection, 
and the time required per operation is always independent of the size of the collection. 


public static void main(String[] args) 

{ // Create a queue and enqueue/dequeue strings. 

Queue<String> q = new Queue<String>(); 

while (! Stdln.isEmptyO) 

{ 

String item = Stdln. readStri ng() ; 
if (litem.equalsf-")) 
q.enqueue(itern); 

else if (Iq.isEmptyO) StdOut. print(q .dequeueO + " "); 

} 

StdOut.println("(" + q.size() + " left on queue)"); 

} 

Test client for Queue 


% more tobe.txt 

to be or not to-be - - that - - - is 

% java Queue < tobe.txt 

to be or not to be (2 left on queue) 
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ALGORITHM 1.3 FIFO queue 


public class Queue<Item> implements Iterable<Item> 

{ 

private Node first; // link to least recently added node 
private Node last; // link to most recently added node 
private int N; // number of items on the queue 

private class Node 

{ // nested class to define nodes 

Item item; 

Node next; 

} 

public boolean isEmptyO { return first == null; } // Or: N == 0. 

public int sizeO { return N; } 

public void enqueue(Item item) 

{ // Add item to the end of the list. 

Node oldlast = last; 

1ast = new Node(); 
last.item = item; 

1ast.next = nul1; 

if (isEmptyO) first = last; 

else oldlast.next = last; 

N++; 

} 

public Item dequeueO 

{ // Remove item from the beginning of the list. 

Item item = first.item; 
fi rst = first.next; 
if (isEmptyO) last = null; 

N—; 

return item; 

} 

// See page 155 for iteratorO implementation. 

// See page 150 for test client main(). 


This generic Queue implementation is based on a linked-list data structure. It can be used to create 
queues containing any type of data. To support iteration, add the highlighted code described for Bag 
on page 155. 
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Trace of Queue development client 
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linked lists are A fundamental alternative to arrays for structuring a collection 
of data. From a historical perspective, this alternative has been available to program¬ 
mers for many decades. Indeed, a landmark in the history of programming languages 
was the development of LISP by John McCarthy in the 1950s, where linked lists are the 
primary structure for programs and data. Programming with linked lists presents all 
sorts of challenges and is notoriously difficult to debug, as you can see in the exercises. 
In modern code, the use of safe pointers, automatic garbage collection (see page 111), and 
ADTs allows us to encapsulate list-processing code in just a few classes such as the ones 
presented here. 
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Bag implementation. Implementing our Bag API using a linked-list data structure is 
simply a matter of changing the name of pushO in Stack to add() and removing the 
implementation of pop(), as shown in algorithm 1.4 on the facing page (doing the 
same for Queue would also be effective but requires a bit more code). This implemen¬ 
tation also highlights the code needed to make Stack, Queue, and Bag all iterable, by 
traversing the list. For Stack the list is in LIFO order; for Queue it is in FIFO order; and 
for Bag it happens to be in LIFO order, but the order is not relevant. As detailed in the 
highlighted code in algorithm 1 . 4 , to implement iteration in a collection, the first step 
is to include 

import java.util.Iterator; 

so that our code can refer to Java’s Iterator interface. The second step is to add 
implements Iterable<Item> 

to the class declaration, a promise to provide an i terator () method. The i terator 0 
method itself simply returns an object from a class that implements the Iterator 
interface: 

public Iterator<Item> iteratorO 
{ return new ListlteratorO; } 

This code is a promise to implement a class that implements the hasNextO, next(), 
and remove () methods that are called when a client uses the foreach construct. To 
implement these methods, the nested class Li stlterator in algorithm 1.4 maintains 
an instance variable current that keeps track of the current node on the list. Then the 
hasNextO method tests if current is null, and the next() method saves a reference 
to the current item, updates current to refer to the next node on the list, and returns 
the saved reference. 
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ALGORITHM 1.4 Bag 


import java.util.Iterator; 

public class Bag<Item> implements Iterable<Item> 

{ 

private Node first; // first node in list 

private class Node 

{ 

Item item; 

Node next; 

} 

public void add(Item item) 

{ // same as push() in Stack 

Node oldfirst = first; 
first = new Node(); 
fi rst.itern = itern; 
first.next = oldfirst; 

} 

public Iterator<Item> iterator() 

{ return new ListlteratorO ; } 

private class Listlterator implements Iterator<Item> 

{ 

private Node current = first; 

public boolean hasNextO 
{ return current != null; } 

public void remove0 { } 
public Item next() 

{ 

Item item = current.item; 
current = current.next; 
return item; 

} 

} 

} 


This Bag implementation maintains a linked list of the items provided in calls to add(). Code for 
isEmptyO and size() is the same as in Stack and is omitted. The iterator traverses the list, main¬ 
taining the current node in current. We can make Stack and Queue iterable by adding the code 
highlighted in red to algorithms i.i and 1 . 2 , because they use the same underlying data structure 
and Stack and Queue maintain the list in LIFO and FIFO order, respectively. 
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Overview The implementations of bags, queues, and stacks that support generics 
and iteration that we have considered in this section provide a level of abstraction that 
allows us to write compact client programs that manipulate collections of objects. De¬ 
tailed understanding of these ADTs is important as an introduction to the study of al¬ 
gorithms and data structures for three reasons. First, we use these data types as building 
blocks in higher-level data structures throughout this book. Second, they illustrate the 
interplay between data structures and algorithms and the challenge of simultaneously 
achieving natural performance goals that may conflict. Third, the focus of several of 
our implementations is on ADTs that support more powerful operations on collections 
of objects, and we use the implementations here as starting points. 

Data structures. We now have two ways to represent collections of objects, arrays and 
linked lists. Arrays are built in to Java; linked lists are easy to build with standard Java 
records. These two alternatives, often referred to as sequential allocation and linked al¬ 
location, are fundamental. Later in the book, we develop ADT implementations that 

combine and extend these basic structures 
in numerous ways. One important exten¬ 
sion is to data structures with multiple 
links. For example, our focus in sections 
3.2 and 3.3 is on data structures known as 
binary trees that are built from nodes that 
each have two links. Another important 
extension is to compose data structures: 
we can have a bag of stacks, a queue of ar¬ 
rays, and so forth. For example, our focus 
in chapter 4 is on graphs, which we rep¬ 
resent as arrays of bags. It is very easy to define data structures of arbitrary complexity 
in this way: one important reason for our focus on abstract data types is an attempt to 
control such complexity. 


data structure 


advantage 


disadvantage 


array 


linked list 


index provides 
immediate access 
to any item 

uses space 
proportional to size 


need to know size 
on initialization 

need reference to 
access an item 


Fundamental data structures 
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our treatment of bags, queues, and stacks in this section is a prototypical ex¬ 
ample of the approach that we use throughout this book to describe data structures 
and algorithms. In approaching a new applications domain, we identify computational 
challenges and use data abstraction to address them, proceeding as follows: 

■ Specify an API. 

■ Develop client code with reference to specific applications. 

■ Describe a data structure (representation of the set of values) that can serve as 
the basis for the instance variables in a class that will implement an ADT that 
meets the specification in the API. 

■ Describe algorithms (approaches to implementing the set of operations) that 
can serve as the basis for implementing the instance methods in the class. 

■ Analyze the performance characteristics of the algorithms. 

In the next section, we consider this last step in detail, as it often dictates which algo¬ 
rithms and implementations can be most useful in addressing real-world applications. 


data structure 

section 

ADT 

representation 

parent-link tree 

1-5 

UnionFi nd 

array of integers 

binary search tree 

3-2, 3-3 

BST 

two links per node 

string 

5-1 

Stri ng 

array, offset, and length 

binary heap 

2.4 

PQ 

array of objects 

hash table 
(separate chaining) 

3-4 

SeparateChainingHashST 

arrays of linked lists 

hash table 
{linear probing) 

3-4 

LinearProbingHashST 

two arrays of objects 

graph adjacency lists 

4 . 1 , 4.2 

Graph 

array of Bag objects 

trie 

5-2 

T ri eST 

node with array of links 

ternary search trie 

5-3 

TST 

three links per node 


Examples of data structures developed in this book 
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| Q&A 

Q. Not all programming languages have generics, even early versions of Java. What are 
the alternatives? 

A. One alternative is to maintain a different implementation for each type of data, as 
mentioned in the text. Another is to build a stack of Object values, then cast to the 
desired type in client code for pop(). The problem with this approach is that type mis¬ 
match errors cannot be detected until run time. But with generics, if you write code to 
push an object of the wrong type on the stack, like this: 

Stack<Apple> stack = new Stack<Apple>(); 

Apple a = new AppleO; 

Orange b = new OrangeO; 

stack.push(a); 

stack.push(b); // compile-time error 

you will get a compile-time error: 

push(Apple) in Stack<Apple> cannot be applied to (Orange) 

This ability to discover such errors at compile time is reason enough to use generics. 

Q. Why does Java disallow generic arrays? 

A. Experts still debate this point. You might need to become one to understand it! For 
starters, learn about covariant arrays and type erasure. 

Q. How do I create an array of stacks of strings? 

A. Use a cast, such as the following: 

Stack<String>[] a = (Stack<String>[]) new StackJN] ; 

Warning: This cast, in client code, is different from the one described on page 134. You 
might have expected to use Object instead of Stack. When using generics, Java 
checks for type safety at compile time, but throws away that information at run time, 
so it is left with Stack<Object>[] or just Stack [], for short, which we must cast to 
Stack<String>[]. 

Q. What happens if my program calls pop() for an empty stack? 
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A. It depends on the implementation. For our implementation on page 149, you will get a 
Nul 1 Poi nterExcept! on. In our implementations on the booksite, we throw a runtime 
exception to help users pinpoint the error. Generally, including as many such checks as 
possible is wise in code that is likely to be used by many people. 

Q. Why do we care about resizing arrays, when we have linked lists? 

A. We will see several examples of ADT implementations that need to use ar¬ 
rays to perform other operations that are not easily supported with linked lists. 
Resi zi ngArrayStack is a model for keeping their memory usage under control. 

Q. Why declare Node as a nested class? Why pri vate? 

A. By declaring the nested class Node to be pri vate, we restrict access to methods and 
instance variables within the enclosing class. One characteristic of a private nested 
class is that its instance variables can be directly accessed from within the enclosing 
class but nowhere else, so there is no need to declare the instance variables publ i c or 
private. Note for experts: A nested class that is not static is known as an inner class, so 
technically our Node classes are inner classes, though the ones that are not generic could 
be static. 

Q. When I type javac Stack, java to run algorithm 1.2 and similar programs, I 
find Stack. cl ass and a file StackSNode. cl ass . What is the purpose of that second 
one? 

A. That file is for the inner class Node. Java’s naming convention is to use $ to separate 
the name of the outer class from the inner class. 

Q. Are there Java libraries for stacks and queues? 

A. Yes and no. Java has a built-in library called java.util .Stack, but you should 
avoid using it when you want a stack. It has several additional operations that are not 
normally associated with a stack, e.g., getting the ith element. It also allows adding an 
element to the bottom of the stack (instead of the top), so it can implement a queue! 
Although having such extra operations may appear to be a bonus, it is actually a curse. 
We use data types not just as libraries of all the operations we can imagine, but also as 
a mechanism to precisely specify the operations we need. The prime benefit of doing so 
is that the system can prevent us from performing operations that we do not actually 
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Q & A (continued) 


want. The j ava. uti 1 . Stack API is an example of a wide interface, which we generally 
strive to avoid. 

Q. Should a client be allowed to insert null items onto a stack or queue? 

A. This question arises frequently when implementing collections in lava. Our imple¬ 
mentation (and Java’s stack and queue libraries) do permit the insertion of null values. 

Q. What should the Stack iterator do if the client calls push () or popO during iterator? 

A. Throw a java, uti 1 . ConcurrentModi fi cation Except! on to make it a fail-fast it¬ 
erator. See 1 . 3 . 50 . 

Q. Can I use a foreach loop with arrays? 

A. Yes (even though arrays do not implement the Iterable interface). The following 
one-liner prints out the command-line arguments: 

public static void main(String[] args) 

{ for (String s : args) StdOut.println(s); } 

Q. Can I use a foreach loop with strings? 

A. No. String does not implement Iterable. 

Q. Why not have a single Col 1 ecti on data type that implements methods to add items, 
remove the most recently inserted, remove the least recently inserted, remove random, 
iterate, return the number of items in the collection, and whatever other operations we 
might desire? Then we could get them all implemented in a single class that could be 
used by many clients. 

A. Again, this is an example of a wide interface. Java has such implementations in its 
java, uti 1 .Array Li st and java, uti 1 . Li nkedLi st classes. One reason to avoid them 
is that it there is no assurance that all operations are implemented efficiently. Through¬ 
out this book, we use APIs as starting points for designing efficient algorithms and data 
structures, which is certainly easier to do for interfaces with just a few operations as 
opposed to an interface with many operations. Another reason to insist on narrow in¬ 
terfaces is that they enforce a certain discipline on client programs, which makes client 
code much easier to understand. If one client uses Stack<Stri ng> and another uses 
Queue<Transacti on>, we have a good idea that the LIFO discipline is important to the 
first and the FIFO discipline is important to the second. 
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EXERCISES 


1.3.1 Add a method i sFul 1 () to Fi xedCapaci tyStackOfStri ngs. 

1.3.2 Give the output printed by j ava Stack for the input 

it was - the best - of times - - - it was - the - - 

1.3.3 Suppose that a client performs an intermixed sequence of (stack) push and pop 
operations. The push operations put the integers 0 through 9 in order onto the stack; 
the pop operations print out the return values. Which of the following sequence(s) 
could not occur? 

a. 4321098765 

b. 4687532901 

c. 2567489310 

d. 4321056789 

e. 1234569870 

f. 0465381729 

g. 1 4 7 9 8 6 5 3 0 2 

h. 2143658790 

1.3.4 Write a stack client Parentheses that reads in a text stream from standard input 
and uses a stack to determine whether its parentheses are properly balanced. For ex¬ 
ample, your program should print true for [()]{}{[() O] 0} and fal se for [(]). 

1.3.5 What does the following code fragment print when N is 50? Give a high-level 
description of what it does when presented with a positive integer N. 

Stack<Integer> stack = new Stack<Integer>() ; 
while (N > 0) 

{ 

stack.push(N % 2); 

N = N / 2; 

} 

for (int d : stack) StdOut.print(d) ; 

StdOut.println(); 

Answer: Prints the binary representation of N (110010 when N is 50). 
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EXERCISES (continued) 


1.3.6 What does the following code fragment do to the queue q? 

Stack<String> stack = new Stack<String>(); 
while (! q. i sEmptyO) 

stack.push(q.dequeueO) ; 
while (Istack.isEmptyO) 
q.enqueue(stack.popO); 

1.3.7 Add a method peek() to Stack that returns the most recently inserted item on 
the stack (without popping it). 

1.3.8 Give the contents and size of the array for Doubl i ngStackOfStri ngs with the 
input 


it was - the best - of times - - - it was - the - - 

1.3.9 Write a program that takes from standard input an expression without left pa¬ 
rentheses and prints the equivalent infix expression with the parentheses inserted. For 
example, given the input: 

l + 2)*3-4)*5-6))) 
your program should print 

((l + 2)*((3-4)*(5-6)) 

1.3.10 Write a filter Infi xToPostfi x that converts an arithmetic expression from in¬ 
fix to postfix. 

1.3.11 Write a program Eval uatePostfi x that takes a postfix expression from stan¬ 
dard input, evaluates it, and prints the value. (Piping the output of your program from 
the previous exercise to this program gives equivalent behavior to Eval uate. 

1.3.12 Write an iterable Stack client that has a static method copyO that takes a stack 
of strings as argument and returns a copy of the stack. Note : This ability is a prime 
example of the value of having an iterator, because it allows development of such func¬ 
tionality without changing the basic API. 

1.3.13 Suppose that a client performs an intermixed sequence of (queue) enqueue and 
dequeue operations. The enqueue operations put the integers 0 through 9 in order onto 
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the queue; the dequeue operations print out the return value. Which of the following 
sequence(s) could not occur? 

a. 0123456789 

b. 4687532901 

c. 2567489310 

d. 4321056789 

1 . 3.14 Develop a class ResizingArrayQueueOfStri ngs that implements the queue 
abstraction with a fixed-size array, and then extend your implementation to use array 
resizing to remove the size restriction. 

1 . 3.15 Write a Queue client that takes a command-line argument k and prints the kth 
from the last string found on standard input (assuming that standard input has k or 
more strings). 

1 . 3.16 Using readlntsO on page 126 as a model, write a static method readDatesO for 
Date that reads dates from standard input in the format specified in the table on page 119 
and returns an array containing them. 

1 . 3.17 Do exercise 1.3.16 for Transacti on. 
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| LINKED-LIST EXERCISES 

This list of exercises is intended to give you experience in working with linked lists. Sugges¬ 
tion: make drawings using the visual representation described in the text. 

1.3.18 Suppose x is a linked-list node and not the last node on the list. What is the ef¬ 
fect of the following code fragment? 

x.next = x.next.next; 

Answer: Deletes from the list the node immediately following x. 

1.3.19 Give a code fragment that removes the last node in a linked list whose first node 
is fi rst. 

1 .3.20 Write a method del ete () that takes an i nt argument k and deletes the kth ele¬ 
ment in a linked list, if it exists. 

1.3.21 Write a method findO that takes a linked list and a string key as arguments 
and returns true if some node in the list has key as its item field, fal se otherwise. 

1.3.22 Suppose that x is a linked list Node. What does the following code fragment do? 

t.next = x.next; 
x.next = t; 

Answer: Inserts node t immediately after node x. 

1.3.23 Why does the following code fragment not do the same thing as in the previous 
question? 

x.next = t; 
t.next = x.next; 

Answer: When it comes time to update t. next, x. next is no longer the original node 
following x, but is instead t itself! 

1.3.24 Write a method removeAfterO that takes a linked-list Node as argument and 
removes the node following the given one (and does nothing if the argument or the next 
field in the argument node is null). 

1.3.25 Write a method i nsertAfterO that takes two linked-list Node arguments and 
inserts the second after the first on its list (and does nothing if either argument is null). 
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1 . 3.26 Write a method remove () that takes a linked list and a string key as arguments 
and removes all of the nodes in the list that have key as its item field. 

1 . 3.27 Write a method max() that takes a reference to the first node in a linked list as 
argument and returns the value of the maximum key in the list. Assume that all keys are 
positive integers, and return 0 if the list is empty. 

1 . 3.28 Develop a recursive solution to the previous question. 

1 . 3.29 Write a Queue implementation that uses a circular linked list, which is the same 
as a linked list except that no links are null and the value of 1 ast. next is f i rst when¬ 
ever the list is not empty. Keep only one Node instance variable (1 ast). 

1 . 3.30 Write a function that takes the first Node in a linked list as argument and (de¬ 
structively) reverses the list, returning the first Node in the result. 

Iterative solution: To accomplish this task, we maintain references to three consecutive 
nodes in the linked list, reverse, fi rst, and second. At each iteration, we extract the 
node fi rst from the original linked list and insert it at the beginning of the reversed 
list. We maintain the invariant that f i rst is the first node of what’s left of the original 
list, second is the second node of what’s left of the original list, and reverse is the first 
node of the resulting reversed list. 

public Node reverse(Node x) 

{ 

Node first = x; 

Node reverse = null; 
while (first != null) 

{ 

Node second = first.next; 
first.next = reverse; 
reverse = first; 
first = second; 

} 

return reverse; 

} 

When writing code involving linked lists, we must always be careful to properly handle 
the exceptional cases (when the linked list is empty, when the list has only one or two 
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LINKED-LIST EXERCISES (continued) 


nodes) and the boundary cases (dealing with the first or last items). This is usually 
much trickier than handling the normal cases. 


Recursive solution: Assuming the linked list has N nodes, we recursively reverse the last 
N— 1 nodes, and then carefully append the first node to the end. 


public Node reverse(Node first) 

{ 

if (first == null) return null; 
if (first.next == null) return first; 
Node second = first.next; 

Node rest = reverse(second); 
second.next = first; 
first.next = nul1 ; 
return rest; 

} 


1.3.31 Implement a nested class DoubleNode for building doubly-linked lists, where 
each node contains a reference to the item preceding it and the item following it in the 
list (null if there is no such item). Then implement static methods for the following 
tasks: insert at the beginning, insert at the end, remove from the beginning, remove 
from the end, insert before a given node, insert after a given node, and remove a given 
node. 
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| CREATIVE PROBLEMS 


1.3.32 Steque. A stack-ended queue or steque is a data type that supports push,pop, and 
enqueue. Articulate an API for this ADT. Develop a linked-list-based implementation. 

1 .3.33 Deque. A double-ended queue or deque (pronounced “deck”) is like a stack or 
a queue but supports adding and removing items at both ends. A deque stores a collec¬ 
tion of items and supports the following API: 


public class Deque<Item> implements Iterable<Item> 


Deque() 

boolean isEmptyO 
int size() 

void pushLeft(Item item) 
void pushRight(Item item) 
Item popLeft() 

Item popRightO 


create an empty deque 
is the deque empty? 
number of items in the deque 
add an item to the left end 
add an item to the right end 
remove an item from the left end 
remove an item from the right end 


API for a generic double-ended queue 


Write a class Deque that uses a doubly-linked list to implement this API and a class 
Resi zi ngArrayDeque that uses a resizing array. 

1.3.34 Random bag. A random bag stores a collection of items and supports the fol¬ 
lowing API: 


public class RandomBag<Item> 

RandomBagO 
boolean isEmptyO 
int size() 
void add(Item item) 

API for, 


implements Iterable<Item> 

create an empty random bag 
is the bag empty? 
number of items in the bag 
add an item 

generic random bag 


Write a class RandomBag that implements this API. Note that this API is the same as for 
Bag, except for the adjective random, which indicates that the iteration should provide 
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CREATIVE PROBLEMS (continued) 


the items in random order (all N\ permutations equally likely, for each iterator). Hint: 
Put the items in an array and randomize their order in the iterator’s constructor. 

1.3.35 Random queue. A random queue stores a collection of items and supports the 
following API: 

public class RandomQueue<Item> 



RandomQueueO 

create an empty random queue 

boolean 

i sEmptyO 

is the queue empty? 

voi d 

enqueue(Item item) 

add an item 

Item 

dequeueO 

remove and return a random item 
(sample without replacement) 

Item 

samp!e() 

return a random item, but do not remove 
(sample with replacement) 


API for a generic random queue 


Write a class RandomQueue that implements this API. Hint: Use an array representation 
(with resizing). To remove an item, swap one at a random position (indexed 0 through 
N-l) with the one at the last position (index N-l). Then delete and return the last ob¬ 
ject, as in Resi zi ngArrayStack. Write a client that deals bridge hands (13 cards each) 
using RandomQueue<Card>. 

1.3.36 Random iterator. Write an iterator for RandomQueue<Item> from the previous 
exercise that returns the items in random order. 

1.3.37 Josephus problem. In the Josephus problem from antiquity, N people are in dire 
straits and agree to the following strategy to reduce the population. They arrange them¬ 
selves in a circle (at positions numbered from 0 to N-l) and proceed around the circle, 
eliminating every Mth person until only one person is left. Legend has it that Josephus 
figured out where to sit to avoid being eliminated. Write a Queue client Josephus that 
takes N and M from the command line and prints out the order in which people are 
eliminated (and thus would show Josephus where to sit in the circle). 

% java Josephus 7 2 

1 3 5 0 4 2 6 
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1.3.38 Delete k th element. Implement a class that supports the following API: 
public class Generali zedQueue<Item> 


General i zedQueueQ 


create an empty queue 
is the queue empty? 
add an item 


boolean isEmptyO 

void insertfltem x) 
Item deletefint k) 


nt k) delete and return the kth least recently inserted item 

API for a generic generalized queue 


First, develop an implementation that uses an array implementation, and then develop 
one that uses a linked-list implementation. Note: the algorithms and data structures 
that we introduce in chapter 3 make it possible to develop an implementation that 
can guarantee that both insertO anddeleteO take time prortional to the logarithm 
of the number of items in the queue—see exercise 3 . 5 . 27 . 

1 .3.39 Ring buffer. A ring buffer, or circular queue, is a FIFO data structure of a fixed 
size N. It is useful for transferring data between asynchronous processes or for storing 
log files. When the buffer is empty, the consumer waits until data is deposited; when the 
buffer is full, the producer waits to deposit data. Develop an API for a Ri ngBuf f e r and 
an implementation that uses an array representation (with circular wrap-around). 

1.3.40 Move-to-front. Read in a sequence of characters from standard input and 
maintain the characters in a linked list with no duplicates. When you read in a previ¬ 
ously unseen character, insert it at the front of the list. When you read in a duplicate 
character, delete it from the list and reinsert it at the beginning. Name your program 
MoveToFront: it implements the well-known move-to-front strategy, which is useful for 
caching, data compression, and many other applications where items that have been 
recently accessed are more likely to be reaccessed. 

1.3.41 Copy a queue. Create a new constructor so that 
Queue<Item> r = new Queue<Item>(q); 

makes r a reference to a new and independent copy of the queue q. You should be able 
to push and pop from either q or r without influencing the other. Hint: Delete all of the 
elements from q and add these elements to both q and r. 
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CREATIVE PROBLEMS (continued) 

1.3.42 Copy a stack. Create a new constructor for the linked-list implementation of 
Stack so that 

Stack<Item> t = new Stack<Item>(s); 
makes t a reference to a new and independent copy of the stack s. 

1.3.43 Listing files. A folder is a list of files and folders. Write a program that takes the 
name of a folder as a command-line argument and prints out all of the files contained 
in that folder, with the contents of each folder recursively listed (indented) under that 
folder’s name. Hint: Use a queue, and see java. i o. Fi 1 e. 

1.3.44 Text editor buffer. Develop a data type for a buffer in a text editor that imple¬ 
ments the following API: 

public class Buffer 


create an empty buffer 

insert c at the cursor position 

delete and return the character at the cursor 

move the cursor k positions to the left 

move the cursor k positions to the right 

number of characters in the buffer 


BufferO 


void insert(char c) 
char deleteO 
void left(int k) 
void right(int k) 
int size() 


API for a text buffer 


Hint: Use two stacks. 

1.3.45 Stack generability. Suppose that we have a sequence of intermixed push and 
pop operations as with our test stack client, where the integers 0, 1, ..., N-l in that 
order ( push directives) are intermixed with N minus signs ( pop directives). Devise an 
algorithm that determines whether the intermixed sequence causes the stack to under¬ 
flow. (You may use only an amount of space independent of N —you cannot store the 
integers in a data structure.) Devise a linear-time algorithm that determines whether a 
given permutation can be generated as output by our test client (depending on where 
the pop directives occur). 
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Solution: The stack does not overflow unless there exists an integer k such that the first 
k pop operations occur before the first k push operations. If a given permutation can be 
generated, it is uniquely generated as follows: if the next integer in the output permuta¬ 
tion is in the top of the stack, pop it; otherwise, push it onto the stack. 

1.3.46 Forbidden triple for stack generability. Prove that a permutation can be gener¬ 
ated by a stack (as in the previous question) if and only if it has no forbidden triple (a, b, 
c ) such that a<b < c with c first, a second, and b third (possibly with other intervening 
integers between c and a and between a and b). 

Partial solution: Suppose that there is a forbidden triple (a, b, c). Item c is popped before 
a and b, but a and b are pushed before c. Thus, when c is pushed, both a and b are on 
the stack. Therefore, a cannot be popped before b. 

1 .3.47 Catenable queues, stacks, or steques. Add an extra operation catenation that (de¬ 
structively) concatenates two queues, stacks, or steques (see exercise 1 . 3 . 32 ). Hint: Use 
a circular linked list, maintaining a pointer to the last item. 

1.3.48 Two stacks with a deque. Implement two stacks with a single deque so that each 
operation takes a constant number of deque operations (see exercise 1 . 3 . 33 ). 

1.3.49 Queue with three stacks. Implement a queue with three stacks so that each 
queue operation takes a constant (worst-case) number of stack operations. Warning: 
high degree of difficulty. 

1.3.50 Fail-fast iterator. Modify the iterator code in Stack to immediately throw a 
java, uti 1 . Concur rentModi fi cati on Exception if the client modifies the collection 
(via push() orpopO) during iteration? b). 

Solution: Maintain a counter that counts the number of push() and pop() operations. 
When creating an iterator, store this value as an Iterator instance variable. Before 
each call to hasNextO and next(), check that this value has not changed since con¬ 
struction of the iterator; if it has, throw the exception. 


www.it-ebooks.info 



1.4 ANALYSIS OF ALGORITHMS 



as people gain experience using computers, they use them to solve difficult prob¬ 
lems or to process large amounts of data and are invariably led to questions like these: 


How long will my program take? 

Why does my program run out of memory? 

You certainly have asked yourself these questions, perhaps when rebuilding a music or 
photo library, installing a new application, working with a large document, or work¬ 
ing with a large amount of experimental data. The questions are much too vague to 
be answered precisely—the answers depend on many factors such as properties of the 
particular computer being used, the particular data being processed, and the particular 
program that is doing the job (which implements some algorithm). All of these factors 
leave us with a daunting amount of information to analyze. 

Despite these challenges, the path to developing useful answers to these basic ques¬ 
tions is often remarkably straightforward, as you will see in this section. This process is 
based on the scientific method, the commonly accepted body of techniques used by sci¬ 
entists to develop knowledge about the natural world. We apply mathematical analysis 
to develop concise models of costs and do experimental studies to validate these models. 

Scientific method The very same approach that scientists use to understand the 
natural world is effective for studying the running time of programs: 

■ Observe some feature of the natural world, generally with precise measurements. 

■ Hypothesize a model that is consistent with the observations. 

■ Predict events using the hypothesis. 

■ Verify the predictions by making further observations. 

■ Validate by repeating until the hypothesis and observations agree. 

One of the key tenets of the scientific method is that the experiments we design must 
be reproducible, so that others can convince themselves of the validity of the hypothesis. 
Hypotheses must also be falsifiable, so that we can know for sure when a given hypoth¬ 
esis is wrong (and thus needs revision). As Einstein famously is reported to have said 
(“No amount of experimentation can ever prove me right; a single experiment can prove 
me wrong"), we can never know for sure that any hypothesis is absolutely correct; we 
can only validate that it is consistent with our observations. 
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Observations Our first challenge is to determine how to make quantitative mea¬ 
surements of the running time of our programs. This task is far easier than in the natu¬ 
ral sciences. We do not have to send a rocket to Mars or kill laboratory animals or split 
an atom—we can simply run the program. Indeed, every time you run a program, you 
are performing a scientific experiment that relates the program to the natural world 
and answers one of our core questions: How long will my program take? 

Our first qualitative observation about most programs is that there is a problem size 
that characterizes the difficulty of the computational task. Normally, the problem size 
is either the size of the input or the value of a command-line argument. Intuitively, the 
running time should increase with problem size, but the question of by how much it 
increases naturally comes up every time we develop and run a program. 

Another qualitative observation for many programs is that the running time is rela¬ 
tively insensitive to the input itself; it depends primarily on the problem size. If this 
relationship does not hold, we need to take steps to better understand and perhaps 
better control the running time’s sensitivity to the input. But it does often hold, so we 
now focus on the goal of better quantifying the relationship between problem size and 
running time. 


Example. As a running example, we will work with the program ThreeSum shown 
here, which counts the number of triples in a file of N integers that sum to 0 (assum¬ 
ing that overflow plays no role). This 
computation may seem contrived to you, 
but it is deeply related to numerous fun¬ 
damental computational tasks (for exam¬ 
ple, see exercise 1 . 4 . 26 ). As a test input, 
consider the file lMints.txt from the 
booksite, which contains 1 million ran¬ 
domly generated i nt values. The second, 
eighth, and tenth entries in lMints.txt 
sum to 0. How many more such triples 
are there in the file? Th reeSum can tell us, 
but can it do so in a reasonable amount 
of time? What is the relationship between 
the problem size N and running time 
for ThreeSum? As a first experiment, try 
running ThreeSum on your computer 
for the files lKints.txt, 2Kints.txt, 

4Kints.txt, and 8 Kints.txt on the 


public class ThreeSum 
{ 

public static int count(int[] a) 

{ // Count triples that sum to 0. 

int N = a.length; 
int cnt = 0; 

for (int i =0; i < N; i++) 

for (int j = i+1; j < N; j++) 
for (int k = j+1; k < N; k++) 
if (a[i] + a[j] + a[k] == 0) 
cnt++; 

return cnt; 

} 

public static void main(String[] args) 

{ 

int[] a = In. readlnts(args[0]) ; 

StdOut.println(count(a)); 

} 


Given N, how long will this program take? 
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% more lMints.txt 
324110 
-442472 
626686 
-157678 
508681 
123414 
-77867 
155091 
129801 
287381 
604242 
686904 
-247109 
77867 
982455 
-210707 
-922943 
-738817 
85168 
855430 


booksite that contain the first 1,000, 2,000, 4,000, and 8,000 integers 
from IMi nts. txt, respectively. You can quickly determine that there are 
70 triples that sum to 0 in IKi nts. txt and that there are 528 triples that 
sum to 0 in 2Kints.txt. The program takes substantially more time 
to determine that there are 4,039 triples that sum to 0 in 4Ki nts. txt, 
and as you wait for the program to finish for 8Ki nts. txt, you will find 
yourself asking the question How long will my program take ? As you will 
see, answering this question for this program turns out to be easy. In¬ 
deed, you can often come up with 
a fairly accurate prediction while 
the program is running. 

Stopwatch. Reliably measuring 
the exact running time of a given 
program can be difficult. Fortu¬ 
nately, we are usually happy with 
estimates. We want to be able to 
distinguish programs that will 
finish in a few seconds or a few 
minutes from those that might 


% java ThreeSum 1000 lKints.txt 



tick tick tick 


70 

% java ThreeSum 2000 2Kints.txt 



tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 


528 


require a few days or a few months or more, and 
we want to know when one program is twice as 
fast as another for the same task. Still, we need 
accurate measurements to generate experimental 
data that we can use to formulate and to check 
the validity of hypotheses about the relationship 
between running time and problem size. For this 
purpose, we use the Stopwatch data type shown 
on the facing page. Its elapsedTimeO method 
returns the elapsed time since it was created, in 
seconds. The implementation is based on using 
the Java system’s currentTimeMillisO method, 
which gives the current time in milliseconds, to 
save the time when the constructor is invoked, 
then uses it again to compute the elapsed time 
when el apsedTi me () is invoked. 


% java ThreeSum 4000 4Kints.txt 



tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 
tick tick tick tick tick tick tick tick 


4039 


tick tick tick tick tick tick tick tick 


Observing the running time of a program 
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API public class Stopwatch 


StopwatchO 
double elapsedTimeQ 


create a stopwatch 

return elapsed time since creation 


typical client public static void main(String[] args) 

{ 

int N = Integer. parselnt(args [0]) ; 

i nt [] a = new int[N]; 

for (int i =0; i < N; i++) 

a[i] = StdRandom.uniform(-1000000, 1000000); 
Stopwatch timer = new StopwatchO; 
int cnt = ThreeSum.count(a); 
double time = timer. elapsedTimeO ; 

StdOut. pri ntl n(cnt + " triples " + time); 


application % j ava stopwatch 1000 

51 triples 0.488 seconds 

% java Stopwatch 2000 
516 triples 3.855 seconds 


implementation public class Stopwatch 

{ 

private final long start; 
public StopwatchO 

{ start = System.currentTimeMi11is(); } 

public double elapsedTimeO 

{ 

long now = System.currentTimeMillisO; 
return (now - start) / 1000.0; 

} 


An abstract data type for a stopwatch 
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Analysis of experimental data. The program Doubl i ngTest on the facing page is a 
more sophisticated Stopwatch client that produces experimental data for Th reeSum. It 
generates a sequence of random input arrays, doubling the array size at each step, and 
prints the running times of Th reeSum. count () for each input size. These experiments 
are certainly reproducible—you can also run them on your own computer, as many 
times as you like. When you run Doubl i ngTest, you will find yourself in a prediction- 
verification cycle: it prints several lines very quickly, but then slows down considerably. 
Each time it prints a line, you find yourself wondering how long it will be until it prints 
the next line. Of course, since you have a different computer from ours, the actual run¬ 
ning times that you get are likely to be different from those shown for our computer. 
Indeed, if your computer is twice as fast as ours, your running times will be about half 
ours, which leads immediately to the well-founded hypothesis that running times on 
different computers are likely to differ by a constant factor. Still, you will find yourself 
asking the more detailed question How long will my program take, as a function of the 
input size? To help answer this question, we plot the data. The diagrams at the bottom of 
the facing page show the result of plotting the data, both on a normal and on a log-log 
scale, with the problem size N on the x-axis and the running time T(N) on the y-axis. 
The log-log plot immediately leads to a hypothesis about the running time—the data 
fits a straight line of slope 3 on the log-log plot. The equation of such a line is 

lg(T(AT)) = 3 lg N + lg a 
(where a is a constant) which is equivalent to 

T(N) = aN 3 

the running time, as a function of the input size, as desired. We can use one of our data 
points to solve for a —for example, T(8000) = 51.1 = «8000 3 , so a = 9.98X10 -11 —and 
then use the equation 

T(N) = 9.98X10~ n N 3 

to predict running times for large N. Informally, we are checking the hypothesis 
that the data points on the log-log plot fall close to this line. Statistical methods are 
available for doing a more careful analysis to find estimates of a and the exponent 
b, but our quick calculations suffice to estimate running time for most purposes. For 
example, we can estimate the running time on our computer for N = 16,000 to be 
about 9.98X10 -11 16000 3 = 408.8 seconds, or about 6.8 minutes (the actual time was 
409.3 seconds). While waiting for your computer to print the line for N = 16,000 in 
Doubl i ngTest, you might use this method to predict when it will finish, then check the 
result by waiting to see if your prediction is true. 
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program to perform experiments results of experiments 

public class DoublingTest 

{ 

public static double timeTrial (int N) 

{ // Time ThreeSum.count() for N random 6-digit ints. 

int MAX = 1000000; 
i nt[] a = new i nt[N] ; 
for (int i =0; i < N; i++) 

a[i] = StdRandom.uniform(-MAX, MAX); 

Stopwatch timer = new Stopwatch(); 
int cnt = ThreeSum.count(a); 
return timer.elapsedTimeO ; 

} 

public static void main(String[] args) 

{ // Print table of running times. 

for (int N = 250; true; N += N) 

{ // Print time for problem size N. 

double time = timeTrial (N) ; 

StdOut.printf("%7d %5.1f\n", N, time); 

} 

} 

} 


% java DoublingTest 
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So far, this process mirrors the process scientists use when trying to understand 
properties of the real world. A straight line in a log-log plot is equivalent to the hy¬ 
pothesis that the data fits the equation T(N) = aN b . Such a fit is known as a power law. 
A great many natural and synthetic phenomena are described by power laws, and it is 
reasonable to hypothesize that the running time of a program does, as well. Indeed, for 
the analysis of algorithms, we have mathematical models that strongly support this and 
similar hypotheses, to which we now turn. 

Mathematical models In the early days of computer science, D. E. Knuth postu¬ 
lated that, despite all of the complicating factors in understanding the running times of 
our programs, it is possible, in principle, to build a mathematical model to describe the 
running time of any program. Knuth’s basic insight is simple: the total running time of 
a program is determined by two primary factors: 

■ The cost of executing each statement 

■ The frequency of execution of each statement 

The former is a property of the computer, the Java compiler and the operating system; 
the latter is a property of the program and the input. If we know both for all instruc¬ 
tions in the program, we can multiply them together and sum for all instructions in the 
program to get the running time. 

The primary challenge is to determine the frequency of execution of the statements. 
Some statements are easy to analyze: for example, the statement that sets cnt to 0 in 
Th reeSum. count () is executed exactly once. Others require higher-level reasoning: for 
example, the i f statement in Th reeSum. count () is executed precisely 

N(N-l)(N-2)/6 

times (the number of ways to pick three different numbers from the input array—see 
exercise 1 . 4 . 1 ). Others depend on the input data: for example the number of times the 
instruction cnt++ in Th reeSum. count () is executed is precisely the number of triples 
that sum to 0 in the input, which could range from 0 of them to all of them. In the case 
of Doubl i ngTest, where we generate the numbers randomly, it is possible to do a prob¬ 
abilistic analysis to determine the expected value of this quantity (see exercise 1 . 4 . 40 ). 

Tilde approximations. Frequency analyses of this sort can lead to complicated and 
lengthy mathematical expressions. For example, consider the count just considered of 
the number of times the i f statement in Th reeSum is executed: 

N(N -1 )(N—2)/6 = N 3 /6 - N 2 /2 + N/3 
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As is typical in such expressions, the terms after 
the leading term are relatively small (for exam¬ 
ple, when N = 1,000 the value of — N 2 / 2 + JV/3 
499,667 is certainly insignificant by com¬ 
parison with N 3 /6 ~ 166,666,667). To allow us 
to ignore insignificant terms and therefore sub¬ 
stantially simplify the mathematical formulas 
that we work with, we often use a mathemati¬ 
cal device known as the tilde notation (~). This 
notation allows us to work with tilde approxi¬ 
mations, where we throw away low-order terms 
that complicate formulas and represent a negli¬ 
gible contribution to values of interest: 


Typical tilde approximations 

For example, we use the approximation ~N 3 /6 to de¬ 
scribe the number of times the i f statement in 
ThreeSum is executed, since N 3 /6 — N 2 / 2 + NI 3 di¬ 
vided by N 3 /6 approaches 1 as N grows. Most of¬ 
ten, we work with tilde approximations of the form 
g(N) ~af(N) where f(N) = N b ( log N) c with a, b, and c 
constants and refer to f(N) as the order of growth of g(N). 
When using the logarithm in the order of growth, we gener¬ 
ally do not specify the base, since the constant a can absorb 
that detail. This usage covers the relatively few functions 
that are commonly encountered in studying the order of 
growth of a program’s running time shown in the table at 
left (with the exception of the exponential, which we defer 
to context). We will describe these functions in more de¬ 
tail and briefly discuss why they appear in the analysis of 
algorithms after we complete our treatment of ThreeSum. 


order of growth 


description 

function 

constant 

1 

logarithmic 

log A 

linear 

A 

linearithmic 

A log A 

quadratic 

N 2 

cubic 

1 V 3 

exponential 

2 n 


Commonly encountered 
order-of-growth functions 


Definition. We write ~f{N) to represent 
any function that, when divided by/(IV), 
approaches 1 as N grows, and we write 
g(N) ~ /(A) to indicate that g(N)/f(N) 
approaches 1 as N grows. 


N 3 1 6 



tilde order 

approximation of growth 


N 3 I6 - N 2 /2 + A/3 
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N 2 /2 - N/2 

~N 2 I2 

A 2 

lglV + 1 
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IgA 

3 
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Approximate running time. To follow through on Knuth’s approach to develop a 
mathematical expression for the total running time of a Java program, we can (in prin¬ 
ciple) study our Java compiler to find the number of machine instructions correspond¬ 
ing to each Java instruction and study our machine specifications to find the time of 
execution of each of the machine instructions, to produce a grand total. This process, 
for Th r eeSum, is briefly summarized on the facing page. We classify blocks of Java state¬ 
ments by their frequency of execution, develop leading-term approximations for the 
frequencies, determine the cost of each statement, and then compute a total. Note that 
some frequencies may depend on the input. In this case, the number of times cnt++ 
is executed certainly depends on the input—it is the number of triples that sum to 0, 
and could range from 0 to ~N 3 / 6. We stop short of exhibiting the details (values of the 
constants) for any particular system, except to highlight that by using constant values t 0 , 
fj, t 2 , ... for the time taken by the blocks of statements, we are assuming that each block 
of Java statements corresponds to machine instructions that require a specified fixed 
amount of time. A key observation from this exercise is to note that only the instruc¬ 
tions that are executed the most frequently play a role in the final total—we refer to 
these instructions as the inner loop of the program. For ThreeSum, the inner loop is the 
statements that increment k and test that it is less than N and the statements that test 
whether the sum of three given numbers is 0 (and possibly the statement that imple¬ 
ments the count, depending on the input). This behavior is typical: the running times 
of a great many programs depend only on a small subset of their instructions. 

Order-of-growth hypothesis. In summary, the experiments on page 177 and the math¬ 
ematical model on page 181 both support the following hypothesis: 

Property A. The order of growth of the running time of Th reeSum (to compute the 
number of triples that sum to 0 among N numbers) is N 3 . 

Evidence: Let T(N) be the running time of ThreeSum for N numbers. The math¬ 
ematical model just described suggests that T(N) ~ aN 3 for some machine-de- 
pendent constant a; experiments on many computers (including yours and ours) 
validate that approximation. 


Throughout this book, we use the term property to refer to a hypothesis that needs to 
be validated through experimentation. The end result of our mathematical analysis is 
precisely the same as the end result of our experimental analysis—the running time 
of ThreeSum is ~ a N 3 for a machine-dependent constant a. This match validates both 
the experiments and the mathematical model and also exhibits more insight about the 
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public class ThreeSum 

{ 

public static int count(int[] a) 

{ 


B- 

C- 

D- 


int N = a.length; 
int cnt = 0; 


for (int i = 0; i < N; i++ ) 


for (int j = i+1; Ij < N; j++ I) 


for (int k = j +1;| k < N; k++|) 


if (a[i] + a[j] + a[k] == 0) 


return cnt; 



public static void main(String[] args) 

inner 

int[] a = In . readlnts(args [0]) ; *°°^ 

StdOut.println(count(a)); 

} 


Anatomy of a program's statement execution frequencies 
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total time 
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Analyzing the running time of a program (example) 
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program because it does not require experimentation to determine the exponent. With 
some effort, we could validate the value of a on a particular system as well, though that 
activity is generally reserved for experts in situations where performance is critical. 

Analysis of algorithms. Hypotheses such as property a are significant because they 
relate the abstract world of a Java program to the real world of a computer running it. 
Working with the order of growth allows us to take one further step: to separate a pro¬ 
gram from the algorithm it implements. The idea that the order of growth of the run¬ 
ning time of Th reeSum is N 3 does not depend on the fact that it is implemented in Java 
or that it is running on your laptop or someone else’s cellphone or a supercomputer; it 
depends primarily on the fact that it examines all the different triples of numbers in the 
input. The algorithm that you are using (and sometimes the input model) determines 
the order of growth. Separating the algorithm from the implementation on a particular 
computer is a powerful concept because it allows us to develop knowledge about the 
performance of algorithms and then apply that knowledge to any computer. For ex¬ 
ample, we might say that Th reeSum is an implementation of the brute-force algorithm 
“compute the sum of all different triples, counting those that sum to 0”—we expect that an 
implementation of this algorithm in any programming language on any computer will 
lead to a running time that is proportional to N 3 . In 
fact, much of the knowledge about the performance 
of classic algorithms was developed decades ago, but 
that knowledge is still relevant to today’s computers. 

Cost model. We focus attention on properties of al¬ 
gorithms by articulating a cost model that defines the 
basic operations used by the algorithms we are study¬ 
ing to solve the problem at hand. For example, an ap¬ 
propriate cost model for the 3-sum problem, shown 
at right, is the number of times we access an array 
entry. With this cost model, we can make precise mathematical statements about prop¬ 
erties of an algorithm, not just a particular implementation, as follows: 

Proposition B. The brute-force 3-sum algorithm uses ~N 3 12 array accesses to 

compute the number of triples that sum to 0 among N numbers. 

Proof: The algorithm accesses each of the 3 numbers for each of the ~N 3 /6 triples. 


3-sum cost model. When 
studying algorithms to 
solve the 3-sum problem, 
we count array accesses 
(the number of times an 
array entry is accessed, for 
read or write). 


We use the term proposition to refer to mathematical truths about algorithms in terms 
of a cost model. Throughout this book, we study the algorithms that we consider within 
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the framework of a specific cost model. Our intent is to articulate cost models such that 
the order of growth of the running time for a given implementation is the same as the 
order of growth of the cost of the underlying algorithm (in other words, the cost model 
should include operations that fall within the inner loop). We seek precise mathemati¬ 
cal results about algorithms (propositions) and also hypotheses about performance 
of implementations (properties) that you can check through experimentation. In this 
case, proposition b is a mathematical truth that supports the hypothesis stated in 
property A, which we have validated with experiments, in accordance with the scien¬ 
tific method. 
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Summary. For many programs, developing a mathematical model of running time 
reduces to the following steps: 

■ Develop an input model, including a definition of the problem size. 

■ Identify the inner loop. 

■ Define a cost model that includes operations in the inner loop. 

■ Determine the frequency of execution of those operations for the given input. 
Doing so might require mathematical analysis —we will consider some examples 
in the context of specific fundamental algorithms later in the book. 

If a program is defined in terms of multiple methods, we normally consider the 
methods separately. As an example, consider our example program of section 1 . 1 , 
BinarySearch. 

Binary search. The input model is the array a[] of size N; the inner loop is the 
statements in the single while loop; the cost model is the compare operation 
(compare the values of two array entries); and the analysis, discussed in section 
1.1 and given in full detail in proposition b in section 3 . 1 , shows that the num¬ 
ber of compares is at most lg N + 1. 

Whitelist. The input model is the N numbers in the whitelist and the M numbers 
on standard input where we assume M » N; the inner loop is the statements in 
the single while loop; the cost model is the compare operation (inherited from 
binary search); and the analysis is immediate given the analysis of binary search— 
the number of compares is at most M (lg N + 1). 

Thus, we draw the conclusion that the order of growth of the running time of the 
whitelist computation is at most M lg N, subject to the following considerations: 

■ If N is small, the input-output cost might dominate. 

■ The number of compares depends on the input—it lies between ~M and ~M 
lg N, depending on how many of the numbers on standard input are in the 
whitelist and on how long the binary search takes to find the ones that are (typi¬ 
cally it is ~MlgN). 

■ We are assuming that the cost of Arrays. sortO is small compared to M lg N. 
Arrays. sortO implements the mergesort algorithm, and in section 2 . 2 , we 
will see that the order of growth of the running time of mergesort is N log N 
(see proposition g in chapter 2 ), so this assumption is justified. 

Thus, the model supports our hypothesis from section 1.1 that the binary search algo¬ 
rithm makes the computation feasible when M and N are large. If we double the length 
of the standard input stream, then we can expect the running time to double; if we 
double the size of the whitelist, then we can expect the running time to increase only 
slightly. 
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developing mathematical models for the analysis of algorithms is a fruitful area 
of research that is somewhat beyond the scope of this book. Still, as you will see with 
binary search, mergesort, and many other algorithms, understanding certain math¬ 
ematical models is critical to understanding the efficiency of fundamental algorithms, 
so we often present details and/or quote the restilts of classic studies. When doing so, we 
encounter various functions and approximations that are widely used in mathemati¬ 
cal analysis. For reference, we summarize some of this information in the tables below. 


description 

notation 

definition 

floor 

w 

largest integer not greater than x 

ceiling 

M 

smallest integer not smaller than x 

natural logarithm 

In N 

log e N (x such that e x = N) 

binary logarithm 

lg N 

log 2 N (x such that 2 X = N) 

integer 

binary logarithm 

LigNj 

largest integer not greater than IgN 
(# bits in binary representation of N) — 1 

harmonic numbers 

h n 

1 + 1/2 + 1/3 + 1/4 + ... + 1/JV 

factorial 

N\ 

1X2X3X4X... X N 


Commonly encountered functions in the analysis of algorithms 


description 

approximation 


harmonic sum 

H n = 1 + 1/2 + 1/3 + 1/4 + ... 

+ 1 IN ~ In N 

triangular sum 

1+2 + 3 + 4 + ... + N 

~ N 2 12 

geometric sum 

1+2 + 4 + 8 + ... + N = 2N-1 

~ 2 N when N = 2" 

Stirling’s 

approximation 

lg N! = lg 1 + lg 2 + lg 3 + lg 4 + . 

.. + lgN~NlgN 

binomial 

coefficients 

(. k^) ~ when k is a small constant 

exponential 

(l-l/x)*~ lie 



Useful approximations for the analysis of algorithms 
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Order-of-growth classifications We use just a few structural primitives (state¬ 
ments, conditionals, loops, nesting, and method calls) to implement algorithms, so very 
often the order of growth of the cost is one of just a few functions of the problem size N. 
These functions are summarized in the table on the facing page, along with the names 
that we use to refer to them, typical code that leads to each function, and examples. 

Constant. A program whose running time’s order of growth is constant executes a 
fixed number of operations to finish its job; consequently its running time does not 
depend on N. Most Java operations take constant time. 

Logarithmic. A program whose running time’s order of growth is logarithmic is barely 
slower than a constant-time program. The classic example of a program whose running 
time is logarithmic in the problem size is binary search (see BinarySearch on page 47). 
The base of the logarithm is not relevant with respect to the order of growth (since all 
logarithms with a constant base are related by a constant factor), so we use log N when 
referring to order of growth. 

Linear. Programs that spend a constant amount of time processing each piece of input 
data, or that are based on a single f o r loop, are quite common. The order of growth of 
such a program is said to be linear —its running time is proportional to N. 

Linearithmic. We use the term linearithmic to describe programs whose running time 
for a problem of size N has order of growth N log N. Again, the base of the logarithm 
is not relevant with respect to the order of growth. The prototypical examples of lin¬ 
earithmic algorithms are Merge. sortO (see algorithm 2 . 4 ) and Quick. sort() (see 
ALGORITHM 2 . 5 ). 

Quadratic. A typical program whose running time has order of growth N 2 has 
two nested for loops, used for some calculation involving all pairs of N elements. 
The elementary sorting algorithms Selection.sortO (see algorithm 2 . 1 ) and 
Insert! on. sort() (see algorithm 2 . 2 ) are prototypes of the programs in this 
classification. 

Cubic. A typical program whose running time has order of growth N 3 has three nested 
for loops, used for some calculation involving all triples of N elements. Our example 
for this section, ThreeSum, is a prototype. 

Exponential. In chapter 6 (but not until then!) we will consider programs whose 
running times are proportional to 2 N or higher. Generally, we use the term exponential 
to refer to algorithms whose order of growth is b N for any constant b > 1, even though 
different values of b lead to vastly different running times. Exponential algorithms are 
extremely slow—you will never run one of them to completion for a large problem. 
Still, exponential algorithms play a critical role in the theory of algorithms because 
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description 


order of 
growth 


typical code framework 


description example 


constant 

1 

a = b + c; 

statement 

add two 
numbers 

logarithmic 

logiV 

[ see page 47 ] 

divide in 
half 

binary 

search 

linear 

N 

doubl e max = a[0]; 
for (int i =1; i < N; i++) 
if (a[i] > max) max = a [i] ; 

loop 

find the 
maximum 

linearithmic 

NlogN 

[ see ALGORITHM 2.4 ] 

divide and 
conquer 

mergesort 



for (int i =0; i < N; i++) 



quadratic 

N 2 

for (int j = i+1; j < N; j++) 
if (a[i] + a [ j ] == 0 ) 

double 

loop 

check all 
pairs 



cnt++; 





for (int i =0; i < N; i++) 





for (int j = i+1; j < N; j++) 


check all 
triples 

cubic 

N 3 

for (int k = j+1; k < N; k++) 
if (a[i ] + a[j] + a[k] == 0 ) 

triple loop 



cnt++; 



exponential 

2 n 

[ see CHAPTER 6 ] 

exhasutive 

search 

check all 
subsets 


Summary of common order-of-growth hypotheses 
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there exists a large class of problems for which it seems that an exponential algorithm 
is the best possible choice. 

these classifications are the most common, but certainly not a complete set. The 
order of growth of an algorithm’s cost might be N 2 log N or N 312 or some similar func- 


standard plot 



log-log plot 



problem size 
Typical orders of growth 


tion. Indeed, the detailed analysis of algorithms 
can require the full gamut of mathematical tools 
that have been developed over the centuries. 

A great many of the algorithms that we con¬ 
sider have straightforward performance charac¬ 
teristics that can be accurately described by one 
of the orders of growth that we have considered. 
Accordingly, we can usually work with specific 
propositions with a cost model, such as mergesort 
uses between V2N lg N and NlgN compares that 
immediately imply hypotheses (properties) such 
as the order of growth of mergesort’s running time 
is linearithmic. For economy, we abbreviate such 
a statement to just say mergesort is linearithmic. 

The plots at left indicate the importance of 
the order of growth in practice. The x-axis is 
the problem size; the y-axis is the running time. 
These charts make plain that quadratic and cubic 
algorithms are not feasible for use on large prob¬ 
lems. As it turns out, several important prob¬ 
lems have natural solutions that are quadratic 
but clever algorithms that are linearithmic. Such 
algorithms (including mergesort) are critically 
important in practice because they enable us to 
address problem sizes far larger than could be 
addressed with quadratic solutions. Naturally, we 
therefore focus in this book on developing loga¬ 
rithmic, linear, and linearithmic algorithms for 
fundamental problems. 
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Designing faster algorithms One of the primary reasons to study the order of 
growth of a program is to help design a faster algorithm to solve the same problem. To 
illustrate this point, we consider next a faster algorithm for the 3-sum problem. How 
can we devise a faster algorithm, before even embarking on the study of algorithms? 
The answer to this question is that we have discussed and used two classic algorithms, 
mergesort and binary search, have introduced the facts that the mergesort is linearith- 
mic and binary search is logarithmic. How can we take advantage of these algorithms 
to solve the 3-sum problem? 

Warmup: 2-sum. Consider the easier problem of determining the number of pairs of 
integers in an input file that sum to 0. To simplify the discussion, assume also that the 
integers are distinct. This problem is easily solved in quadratic time by deleting the k 
loop and a[k] from ThreeSum. count (), leaving a double loop that examines all pairs, 
as shown in the quadratic entry in the table on page 187 (we refer to such an implementa¬ 
tion as TwoSum). The implementation below shows how mergesort and binary search 
(see page 47) can serve as a basis for a linearithmic solution to the 2-sum problem. The 
improved algorithm is based on the fact that an entry a [i ] is one of a pair that sums to 
0 if and only if the value -a [i ] is in the array (and a [i ] is not zero). To solve the prob¬ 
lem, we sort the array (to enable binary search) and then, for every entry a [i ] in the ar¬ 
ray, do a binary search for -a [i ] with rankf) in Bi narySearch. If the result is an index 
j with j > i, we increment the count. 

This succinct test covers three cases: 

■ An unsuccessful binary search re¬ 
turns -1, so we do not increment 
the count. 

■ If the binary search re¬ 
turns j > i, we have 
a[i] + a [ j ] = 0, so we incre¬ 
ment the count. 

■ If the binary search returns j 
between 0 and i, we also have 
a[i] + a [ j ] = 0 but do not 
increment the count, to avoid 
double counting. 

The result of the computation is precise¬ 
ly the same as the result of the quadratic 
algorithm, but it takes much less time. 

The running time of the mergesort is Linearithmic solution to the 2-sum problem 


import java.util.Arrays; 

public class TwoSumFast 
{ 

public static int count(int[] a) 

{ // Count pairs that sum to 0. 

Arrays.sort(a); 
int N = a.length; 
int cnt = 0; 

for (int i =0; i < N; i++) 

if (BinarySearch. rank(-a[i] , a) > i) 
cnt++; 

return cnt; 

} 

public static void main(String[] args) 

{ 

int[] a = In. readlnts(args[0]) ; 
StdOut.println(count(a)) ; 

} 

} 
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proportional to N log N, and the N binary searches each take time proportional to log 
N, so the running time of the whole algorithm is proportional to N log N. Developing 
a faster algorithm like this is not merely an academic exercise—the faster algorithm 
enables us to address much larger problems. For example, you are likely to be able to 
solve the 2-sum problem for 1 million integers (lMi nts. txt) in a reasonable amount 
of time on your computer, but you would have to wait quite a long time to do it with 
the quadratic algorithm (see exercise 1 . 4 . 41 ). 

Fast algorithm for 3-sum. The very same idea is effective for the 3-sum problem. 
Again, assume also that the integers are distinct. A pair a [i ] and a [ j ] is part of a triple 
that sums to 0 if and only if the value - (a [i ] + a [ j ]) is in the array (and not a [i ] or 
a [ j ]). The code below sorts the array, then does N (N— 1)/2 binary searches that each 
take time proportional to log N, for a total running time proportional to N 2 log N. Note 
that in this case the cost of the sort is insignificant. Again, this solution enables us to ad¬ 
dress much larger problems (see exercise 1 . 4 . 42 ). The plots in the figure at the bottom 
of the next page show the disparity in costs among these four algorithms for problem 
sizes in the range we have considered. Such differences certainly motivate the search for 
faster algorithms. 

Lower bounds. The table on page 191 summarizes the discussion of this section. An in¬ 
teresting question immediately arises: Can we find algorithms for the 2-sum and 3-sum 

problems that are substantially 
faster than TwoSumFast and 
ThreeSumFast? Is there a linear 
algorithm for 2 -sum or a linea- 
rithmic algorithm for 3-sum? 
The answer to this question is no 
for 2 -sum (under a model that 
counts and allows only compari¬ 
sons of linear or quadratic func¬ 
tions of the numbers) and no one 
knows for 3-sum, though experts 
believe that the best possible al¬ 
gorithm for 3-sum is quadratic. 
The idea of a lower bound on the 
order of growth of the worst-case 
running time for all possible al¬ 
gorithms to solve a problem is a 
very powerful one, which we will 


import java.uti1.Arrays; 

public class ThreeSumFast 

{ 

public static int count(int[] a) 

{ // Count triples that sum to 0. 

Arrays.sort(a); 
int N = a.length; 
int cnt = 0; 

for (int i =0; i < N; i++) 

for (int j = i+1; j < N; j++) 

if (BinarySearch. rank(-a[i]-a[j], a) > j) 
cnt++; 

return cnt; 

} 

public static void main(String[] args) 

{ 

int[] a = In . readlnts(args[0]) ; 

StdOut.println(count(a)); 

} 

} 

N 2 Ig N solution to the 3-sum problem 
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revisit in detail in section 2.2 in the context of sorting. Non¬ 
trivial lower bounds are difficult to establish, but very helpful 
in guiding our search for efficient algorithms. 


algorithm 


order of growth 
of running time 


TwoSum N 2 


TwoSumFast 

ThreeSum 


NlogN 
N 3 

N 2 log N 

Summary of running times 


ThreeSumFast 


the examples in this section set the stage for our treat¬ 
ment of algorithms in this book. Throughout the book, our 
strategy for addressing new problems is the following: 

■ Implement and analyze a straighforward solution to 
the problem. We usually refer to such solutions, like 
ThreeSum and TwoSum, as the brute-force solution. 

■ Examine algorithmic improvements, usually designed 

to reduce the order of growth of the running time, such as TwoSumFast and 
ThreeSumFast. 

■ Run experiments to validate the hypotheses that the new algorithms are faster. 

In many cases, we examine several algorithms for the same problem, because running 
time is only one consideration when choosing an algorithm for a practical problem. We 
will develop this idea in detail in the context of fundamental problems throughout the 
book. 




Costs of algorithms to solve the 2-sum and 3-sum problems 
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Doubling ratio experiments The following is a simple and effective shortcut for 
predicting performance and for determining the approximate order of growth of the 
running time of any program: 

■ Develop an input generator that produces inputs that model the inputs expected 
in practice (such as the random integers in timeTri al () in Doubl i ngTest. 

■ Run the program Doubl i ngRati o given below, a modification of Doubl i ngTest 
that calculates the ratio of each running time with the previous. 

■ Run until the ratios approach a limit 2 b . 

This test is not effective if the ratios do not approach a limiting value, but they do for 
many, many programs, implying the following conclusions: 

■ The order of growth of the running time is approximately N b . 

■ To predict running times, multiply the last observed running time by 2 b and 
double N, continuing as long as desired. If you want to predict for an input size 
that is not a power of 2 times N, you can adjust ratios accordingly (see exercise 
1 . 4 . 9 ). 

As illustrated below, the ratio for ThreeSum is about 8 and we can predict the running 
times for N = 16,000, 32,000, 64,000 to be 408.8, 3270.4, 26163.2 seconds, respectively, 
just by successively multiplying the last time for 8,000 (51.1) by 8. 


program to perform experiments 


public class DoublingRatio 

{ 

public static double timeTrial (int N) 

// same as for DoublingTest (page 177) 

public static void main(String[] args) 

{ 

double prev = timeTrial (125) ; 

for (int N = 250; true; N += N) 

{ 

double time = timeTrial (N) ; 

StdOut.printf("%6d %7.1f ", N, time); 

StdOut.printf("%5.lf\n", time/prev); 
prev = time; 

} 

} 

} 


results of experiments 


% java 

DoublingRati 0 

250 

0.0 

2.7 

500 

0.0 

4.8 

1000 

0.1 

6.9 

2000 

0.8 

7.7 

4000 

6.4 

8.0 

8000 

51.1 

8.0 

predictions 

16000 

408.8 

8.0 

52000 

5270.4 

8.0 

64000 

26165.2 

8.0 
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This test is roughly equivalent to the process described on page 176 (run experiments, 
plot values on a log-log plot to develop the hypothesis that the running time is aN b , 
determine the value of b from the slope of the line, then solve for a), but it is sim¬ 
pler to apply. Indeed, you can accurately predict preformance by hand when you run 
Doubl i ngRati o. As the ratio approaches a limit, just multiply by that ratio to fill in later 
values in the table. Your approximate model of the order of growth is a power law with 
the binary logarithm of that ratio as the power. 

Why does the ratio approach a constant? A simple mathematical calculation shows 
that to be the case for all of the common orders of growth just discussed (except 
exponential): 

Proposition C. (Doubling ratio) IfT(N) ~ aNMgiVthen T(2N)/T(N) ~ 2 b . 

Proof: Immediate from the following calculation: 

T(2N)/T(N) = a(2N) b lg(2N)/aN b \gN 
= 2 b (1 + lg2 / lgiV) 

~2 b 


Generally, the logarithmic factor cannot be ignored when developing a mathematical 
model, but it plays a less important role in predicting performance with a doubling 
hypothesis. 

you should consider running doubling ratio experiments for every program that 
you write where performance matters—doing so is a very simple way to estimate the 
order of growth of the running time, perhaps revealing a performance bug where a 
program may turn out to be not as efficient as you might think. More generally, we can 
use hypotheses about the order of growth of the running time of programs to predict 
performance in one of the following ways: 

Estimating the feasibility of solving large problems. You need to be able to answer 
this basic question for every program that you write: Will the program be able to process 
this given input data in a reasonable amount of time? To address such questions for a 
large amount of data, we extrapolate by a much larger factor than for doubling, say 10, 
as shown in the fourth column in the table at the bottom of the next page. Whether it is 
an investment banker running daily financial models or a scientist running a program 
to analyze experimental data or an engineer running simulations to test a design, it is 
not unusual for people to regularly run programs that take several hours to complete, 
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so the table focuses on that situation. Knowing the order of growth of the running time 
of an algorithm provides precisely the information that you need to understand limita¬ 
tions on the size of the problems that you can solve. Developing such understanding is 
the most important reason to study performance. Without it, you are likely have no idea 
how much time a program will consume; with it, you can make a back-of-the-envelope 
calculation to estimate costs and proceed accordingly. 

Estimating the value of using a faster computer. You also may be faced with this basic 
question, periodically: How much faster can I solve the problem if I get a faster computer? 
Generally, if the new computer is x times faster than the old one, you can improve your 
running time by a factor of x. But it is usually the case that you can address larger prob¬ 
lems with your new computer. How will that change affect the running time? Again, the 
order of growth is precisely the information needed to answer that question. 

A famous rule of thumb known as Moore’s Law implies that you can expect to have a 
computer with about double the speed and double the memory 18 months from now, 
or a computer with about 10 times the speed and 10 times the memory in about 5 years. 
The table below demonstrates that you cannot keep pace with Moore’s Law if you are 
using a quadratic or a cubic algorithm, and you can quickly determine whether that is 
the case by doing a doubling ratio test and checking that the ratio of running times as 
the input size doubles approaches 2, not 4 or 8 . 


order of growth of time for a program that takes a few hours for input of size N 


description 

function 

2x 

factor 

10x 

factor 

pred icted ti me for 10N 

predicted time forlO/V 
on a 10x faster computer 

linear 

N 

2 

10 

a day 

a few hours 

linearithmic 

IVlogJV 

2 

10 

a day 

a few hours 

quadratic 

N 2 

4 

100 

a few weeks 

a day 

cubic 

N 3 

8 

1,000 

several months 

a few weeks 

exponential 

2 N 

2 N 

2 9N 

never 

never 


Predictions on the basis of order-of-growth function 
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Caveats There are many reasons that you might get inconsistent or misleading re¬ 
sults when trying to analyze program performance in detail. All of them have to do with 
the idea that one or more of the basic assumptions underlying our hypotheses might be 
not quite correct. We can develop new hypotheses based on new assumptions, but the 
more details that we need to take into account, the more care is required in the analysis. 

Large constants. With leading-term approximations, we ignore constant coefficients 
in lower-order terms, which may not be justifed. For example, when we approximate 
the function 2 N 2 + cN by ~2 N 2 , we are assuming that c is small. If that is not the case 
(suppose that c is 10 3 or 10 6 ) the approximation is misleading. Thus, we have to be 
sensitive to the possibility of large constants. 

Nondominant inner loop. The assumption that the inner loop dominates may not 
always be correct. The cost model might miss the true inner loop, or the problem size 
N might not be sufficiently large to make the leading term in the mathematical descrip¬ 
tion of the frequency of execution of instructions in the inner loop so much larger 
than lower-order terms that we can ignore them. Some programs have a significant 
amount of code outside the inner loop that needs to be taken into consideration. In 
other words, the cost model may need to be refined. 

Instruction time. The assumption that each instruction always takes the same amount 
of time is not always correct. For example, most modern computer systems use a tech¬ 
nique known as caching to organize memory, in which case accessing elements in huge 
arrays can take much longer if they are not close together in the array. You might ob¬ 
serve the effect of caching for ThreeSum by letting Doubl i ngTest run for a while. After 
seeming to converge to 8, the ratio of running times may jump to a larger value for large 
arrays because of caching. 

System considerations. Typically, there are many, many things going on in your com¬ 
puter. Java is one application of many competing for resources, and Java itself has many 
options and controls that significantly affect performance. A garbage collector or a just- 
in-time compiler or a download from the internet might drastically affect the results 
of experiments. Such considerations can interfere with the bedrock principle of the 
scientific method that experiments should be reproducible, since what is happening at 
this moment in your computer will never be reproduced again. Whatever else is going 
on in your system should in principle be negligible or possible to control. 

Too close to call. Often, when we compare two different programs for the same task, 
one might be faster in some situations, and slower in others. One or more of the consid¬ 
erations just mentioned could make the difference. There is a natural tendency among 
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some programmers (and some students) to devote an extreme amount of energy run¬ 
ning races to find the “best” implementation, but such work is best left for experts. 

Strong dependence on inputs. One of the first assumptions that we made in order to 
determine the order of growth of the program’s running time of a program was that the 
running time should be relatively insensitive to the inputs. When that is not the case, we 
may get inconsistent results or be unable to validate our hypotheses. For example, sup¬ 
pose that we modify Th reeSum to answer the question Does the input have a triple that 
sums to 0 ? by changing it to return a bool ean value, replacing cnt++ by return true 
and adding return fal se as the last statement. The order of growth of the running 
time of this program is constant if the first three integers sum to 0 and cubic if there are 
no such triples in the input. 

Multiple problem parameters. We have been focusing on measuring performance as a 
function of a single parameter, generally the value of a command-line argument or the 
size of the input. However, it is not unusual to have several parameters. A typical ex¬ 
ample arises when an algorithm involves building a data structure and then performing 
a sequence of operations that use that data structure. Both the size of the data structure 
and the number of operations are parameters for such applications. We have already 
seen an example of this in our analysis of the problem of whitelisting using binary 
search, where we have N numbers in the whitelist and M numbers on standard input 
and a typical running time proportional to M log N. 

Despite all these caveats, understanding the order of growth of the running time of 
each program is valuable knowledge for any programmer, and the methods that we 
have described are powerful and broadly applicable. Knuth’s insight was that we can 
carry these methods through to the last detail in principle to make detailed, accurate 
predictions. Typical computer systems are extremely complex and close analysis is best 
left for experts, but the same methods are effective for developing approximate esti¬ 
mates of the running time of any program. A rocket scientist needs to have some idea 
of whether a test flight will land in the ocean or in a city; a medical researcher needs to 
know whether a drug trial will kill or cure all the subjects; and any scientist or engineer 
using a computer program needs to have some idea of whether it will run for a second 
or for a year. 
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Coping with dependence on inputs For many problems, one of the most sig¬ 
nificant of the caveats just mentioned is the dependence on inputs, because running 
times can vary widely. The running time of the modification of Th reeSum mentioned 
on the facing page ranges from constant to cubic, depending on the input, so a closer 
analysis is required if we want to predict performance. We briefly consider here some of 
the approaches that are effective and that we will consider for specific algorithms later 
in the book. 

Input models. One approach is to more carefully model the kind of input to be pro¬ 
cessed in the problems that we need to solve. For example, we might assume that the 
numbers in the input to Th reeSum are random i nt values. This approach is challenging 
for two reasons: 

■ The model may be unrealistic. 

■ The analysis may be extremely difficult, requiring mathematical skills quite be¬ 
yond those of the typical student or programmer. 

The first of these is the more significant, often because the goal of a computation is to 
discover characteristics of the input. For example, if we are writing a program to process 
a genome, how can we estimate its performance on a different genome? A good model 
describing the genomes found in nature is precisely what scientists seek, so estimating 
the running time of our programs on data found in nature actually amounts to con¬ 
tributing to that model! The second challenge leads to a focus on mathematical results 
only for our most important algorithms. We will see several examples where a simple 
and tractable input model, in conjunction with classical mathematical analysis, helps 
us predict performance. 

Worst-case performance guarantees. Some applications demand that the running 
time of a program be less than a certain bound, no matter what the input. To provide 
such performance guarantees, theoreticians take an extremely pessimistic view of the 
performance of algorithms: what would the running time be in the worst casd For 
example, such a conservative approach might be appropriate for the software that runs 
a nuclear reactor or a pacemaker or the brakes in your car. We want to guarantee that 
such software completes its job within the bounds that we set because the result could 
be catastrophic if it does not. Scientists normally do not contemplate the worst case 
when studying the natural world: in biology, the worst case might be the extinction 
of the human race; in physics, the worst case might be the end of the universe. But the 
worst case can be a very real concern in computer systems, where the input may be 
generated by another (potentially malicious) user, rather than by nature. For example, 
websites that do not use algorithms with performance guarantees are subject to denial- 
of-service attacks, where hackers flood them with pathological requests that make them 
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run much more slowly than planned. Accordingly, many of our algorithms are designed 
to provide performance guarantees, such as the following: 

Proposition D. In the linked-list implementations of Bag (algorithm 1 . 4 ), Stack 
(algorithm 1 . 2 ), and Queue (algorithm 1 . 3 ), all operations take constant time 
in the worst case. 

Proof: Immediate from the code. The number of instructions executed for each 
operation is bounded by a small constant. Caveat: This argument depends upon 
the (reasonable) assumption that the lava system creates a new Node in constant 
time. 


Randomized algorithms. One important way to provide a performance guarantee is 
to introduce randomness. For example, the quicksort algorithm for sorting that we 
study in section 2.3 (perhaps the most widely used sorting algorithm) is quadratic in 
the worst case, but randomly ordering the input gives a probabilistic guarantee that its 
running time is linearithmic. Every time you run the algorithm, it will take a different 
amount of time, but the chance that the time will not be linearithmic is so small as to be 
negligible. Similarly, the hashing algorithms for symbol tables that we study in section 
3.4 (again, perhaps the most widely used approach) are linear-time in the worst case, 
but constant-time under a probabilistic guarantee. These guarantees are not absolute, 
but the chance that they are invalid is less than the chance your computer will be struck 
by lightning. Thus, such guarantees are as useful in practice as worst-case guarantees. 

Sequences of operations. For many applications, the algorithm “input” might be 
not just data, but the sequence of operations performed by the client. For example, a 
pushdown stack where the client pushes N values, then pops them all, may have quite 
different performance characteristics from one where the client issues an alternating 
sequence N of push and pop operations. Our analysis has to take both situations into 
account (or to include a reasonable model of the sequence of operations). 

Amortized analysis. Accordingly, another way to provide a performance guarantee is 
to amortize the cost, by keeping track of the total cost of all operations, divided by the 
number of operations. In this setting, we can allow some expensive operations, while 
keeping the average cost of operations low. The prototypical example of this type of 
analysis is the study of the resizing array data structure for Stack that we considered in 
section 1.3 (algorithm i.i onpage 141). For simplicity, suppose that Nis a power of 2. 
Starting with an empty structure, how many array entries are accessed for N consecu¬ 
tive calls to pushQ? This quantity is easy to calculate: the number of array accesses is 
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N + 4 + 8 + 16 + ... + 2N = 5N - 4 

The first term accounts for the array access 
within each of the N calls to push(); the sub¬ 
sequent terms account for the array accesses to 
initialize the data structure each time it doubles 
in size. Thus the average number of array access¬ 
es per operation is constant, even though the last 
operation takes linear time. This is known as an 
“amortized” analysis because we spread the cost 
of the few expensive operations, by assigning a 
portion of it to each of a large number of inexpensive operations. Vi sualAccumul ator 
provides an easy way to illustrate the process, shown above. 

Proposition E. In the resizing array implementation of Stack (algorithm i.i), 
the average number of array accesses for any sequence of operations starting from 
an empty data structure is constant in the worst case. 

Proof sketch: For each pushO that causes the array to grow ( say from size N to 
size 2 N), consider the NI2 — 1 pushO operations that most recently caused the 
stack size to grow to k, for k from Nl2 + 2 to N. Averaging the 4 N array accesses to 
grow the array with Nl2 array accesses (one for each push), we get an average cost 
of 9 array accesses per operation. Proving that the number of array accesses used by 
any sequence of M operations is proportional to M is more intricate (see exercise 
1-4-32) 


256-. 


one gray dot 
for each operation 




128 


64 


red dots give cumulative average 5 
_._\__ / 


number of add() operations 


128 


Amortized cost of adding to a RandomBag 


This kind of analysis is widely applicable. In particular, we use resizing arrays as the 
underlying data structure for several algorithms that we consider later in this book. 

it is the task of the algorithm analyst to discover as much relevant information 
about an algorithm as possible, and it is the task of the applications programmer to 
apply that knowledge to develop programs that effectively solve the problems at hand. 
Ideally, we want algorithms that lead to clear and compact code that provides both a 
good guarantee and good performance on input values of interest. Many of the classic 
algorithms that we consider in this chapter are important for a broad variety of ap¬ 
plications precisely because they have these properties. Using them as models, you can 
develop good solutions yourself for typical problems that you face while programming. 
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type 


bytes 


boolean 
byte 
char 
int 
float 
long 
doubl e 


Memory As with running time, a program’s memory usage connects directly to the 
physical world: a substantial amount of your computer’s circuitry enables your pro¬ 
gram to store values and later retrieve them. The more values you need to have stored 
at any given instant, the more circuitry you need. You probably are aware of limits on 
memory usage on your computer (even more so than for time) because you probably 
have paid extra money to get more memory. 

Memory usage is well-defined for Java on your computer (every value requires pre¬ 
cisely the same amount of memory each time that you run your program), but Java is 
implemented on a very wide range of computational devices, and memory consump¬ 
tion is implementation-dependent. For economy, we use the word typical to signal that 
values are subject to machine dependencies. 

One of Java’s most significant features is its memory allocation system, 
which is supposed to relieve you from having to worry about memory. 
Certainly, you are well-advised to take advantage of this feature when ap¬ 
propriate. Still, it is your responsibility to know, at least approximately, 
when a program’s memory requirements will prevent you from solving a 
given problem. 

Analyzing memory usage is much easier than analyzing running time, 
primarily because not as many program statements are involved (just dec¬ 
larations) and because the analysis reduces complex objects to the primi¬ 
tive types, whose memory usage is well-defined and simple to understand: 
we can count up the number of variables and weight them by the number 
of bytes according to their type. For example, since the Java i nt data type 
is the set of integer values between—2,147,483,648 and 2,147,483,647, a 
grand total of 2 32 different values, typical Java implementations use 32 bits 
to represent i nt values. Similar considerations hold for other primitive types: typical 
Java implementations use 8-bit bytes, representing each char value with 2 bytes (16 
bits), each int value with 4 bytes (32 bits), each double and each long value with 8 
bytes (64 bits), and each boolean value with 1 byte (since computers typically access 
memory one byte at a time). Combined with knowledge of the amount of memory 
available, you can calculate limitations from these values. For example, if you have 1GB 
of memory on your computer (1 billion bytes), you cannot fit more than about 32 mil¬ 
lion i nt values or 16 million doubl e values in memory at any one time. 

On the other hand, analyzing memory usage is subject to various differences in ma¬ 
chine hardware and in Java implementations, so you should consider the specific ex¬ 
amples that we give as indicative of how you might go about determining memory 
usage when warranted, not the final word for your computer. For example, many data 
structures involve representation of machine addresses, and the amount of memory 


Typical memory 
requirements for 
primitive types 
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needed for a machine address varies from machine to 
machine. For consistency, we assume that 8 bytes are 
needed to represent addresses, as is typical for 64-bit 
architectures that are now widely used, recognizing 
that many older machines use a 32-bit architecture that 
would involve just 4 bytes per machine address. 

Objects. To determine the memory usage of an object, 
we add the amount of memory used by each instance 
variable to the overhead associated with each object, 
typically 16 bytes. The overhead includes a reference to 
the object’s class, garbage collection information, and 
synchronization information. Moreover, the memory 
usage is typically padded to be a multiple of 8 bytes 
(machine words, on a 64-bit machine). For example, 
an Integer object uses 24 bytes (16 bytes of overhead, 
4 bytes for its i nt instance variable, and 4 bytes of 
padding). Similarly, a Date (page 91) object also uses 32 
bytes: 16 bytes of overhead, 4 bytes for each of its three 
int instance variables, and 4 bytes of padding. A ref¬ 
erence to an object typically is a memory address and 
thus uses 8 bytes of memory. For example, a Counter 
(page 89) object uses 32 bytes: 16 bytes of overhead, 8 
bytes for its String instance variable (a reference), 4 
bytes for its i nt instance variable, and 4 bytes of pad¬ 
ding. When we account for the memory for a reference, 
we account separately for the memory for the object 
itself, so this total does not count the memory for the 
Stri ng value. 


integer wrapper object 

public class Integer 
{ 

private int x; 

}" 


date object 

public class Date 
{ 

private int day; 
private int month; 
private int year; 

}” 


counter object 

public class Counter 
{ 

private String name; 
private int count; 

}" 


node object ( inner class ) 
public class Node 
{ 

private Item item; 
private Node next; 

} 


24 bytes 



Linked lists. A nested non-static (inner) class such Typical object memory requirements 

as our Node class (page 142) requires an extra 8 bytes of 

overhead (for a reference to the enclosing instance). Thus, a Node object uses 40 bytes 
(16 bytes of object overhead, 8 bytes each for the references to the Item and Node ob¬ 
jects, and 8 bytes for the extra overhead). Thus, since an Integer object uses 24 bytes, a 
stack with N integers built with a linked-list representation (algorithm 1 . 2 ) uses 32 + 

64N bytes, the usual 16 for object overhead for Stack, 8 for its reference instance vari¬ 
able, 4 for its i nt instance variable, 4 for padding, and 64 for each entry, 40 for a Node 
and 24 for an Integer. 
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Arrays. Typical memory requirements for various types of arrays in Java are summa¬ 
rized in the diagrams on the facing page. Arrays in Java are implemented as objects, 
typically with extra overhead for the length. An array of primitive-type values typically 
requires 24 bytes of header information (16 bytes of object overhead, 4 bytes for the 
length, and 4 bytes of padding) plus the memory needed to store the values. For ex¬ 
ample, an array of N int values uses 24 + 4A/ bytes (rounded up to be a multiple of 
8 ), and an array of N doubl e values uses 24 + 8N bytes. An array of objects is an array 
of references to the objects, so we need to add the space for the references to the space 
required for the objects. For example, an array of N Date objects (page 91) uses 24 bytes 
(array overhead) plus 8N bytes (references) plus 32 bytes for each object and 4 bytes of 
padding, for a grand total of 24 + 40N bytes. A two-dimensional array is an array of ar¬ 
rays (each array is an object). For example, a two-dimensional M-by-N array of doubl e 
values uses 24 bytes (overhead for the array of arrays) plus 8 Mbytes (references to the 
row arrays) plus M times 16 bytes (overhead from the row arrays) plus M times N times 
8 bytes (for the N doubl e values in each of the M rows) for a grand total of 8 NM + 
32 M + 24 ~ 8NM bytes. When array entries are objects, a similar accounting leads to a 
total of 8 NM + 32 M + 24 ~ 8NM bytes for the array of arrays filled with references to 
objects, plus the memory for the objects themselves. 

String objects. We account for memory in Java’s Stri ng objects in the same way as 
for any other object, except that aliasing is common for strings. The standard Stri ng 
implementation has four instance variables: a reference to a character array (8 bytes) 
and three int values (4 bytes each). The first int value is an offset into the character ar¬ 
ray; the second is a count (the string length). In terms of the instance variable names in 
the drawing on the facing page, the string that is represented consists of the characters 
valuejoffset] through val ue [offset + count - 1]. The third i nt value in Stri ng 
objects is a hash code that saves recomputation in certain circumstances that need not 
concern us now. Therefore, each String object uses a total of 40 bytes (16 bytes for 
object overhead plus 4 bytes for each of the three int instance variables plus 8 bytes for 
the array reference plus 4 bytes of padding). This space requirement is in addition to 
the space needed for the characters themselves, which are in the array. The space needed 
for the characters is accounted for separately because the char array is often shared 
among strings. Since Stri ng objects are immutable, this arrangement allows the imple¬ 
mentation to save memory when String objects have the same underlying val ue []. 

String values and substrings. A Stri ng of length N typically uses 40 bytes (for the 
String object) plus 24 + 2N bytes (for the array that contains the characters) for a 
total of 64 + 2N bytes. But it is typical in string processing to work with substrings, and 
Java’s representation is meant to allow us to do so without having to make copies of 
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array of int values 

i nt [] a = new i nt [N]; 


array of doubl e values 

doublet] c = new double[N]; 


a 


int value y 
(4 bytes) 


object 

overhead 


N 

padding 


Total: 24 + 4 N (N even ) 


— 16 bytes 



N i nt values 
(4 N bytes) 


array of objects 



int[] -AN 


doublet] ~8 N 

Date[] ~40N 

doubled [] ~8 NM 


i nt value ^ 
(4 bytes ) 


object 

overhead 


N 


padding 


-16 bytes 


^ Ndoubl e values 
/ (8 N bytes) 


Total: 24 + 8 N 


array of arrays (two-dimensional array) 

doublet] [] t; 
t = new doubletM]tN]; 


16 bytt 



M reference: 
(8 M bytes) 



N 


padding 


- 4 bytes 


Total: 24 + 8M + MX (24 + 8N ) = 24 + 32 M + 8MN 


Typical memory requirements for arrays of i nt values, doubl e values, objects, and arrays 
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CHAPTER 1 


Fundamentals 


String object (Java library) 

public class String 
{ 

private char[] value; 
private int offset; 
private int count; 
private int hash; 


} 



substring example 

String genome = "CGCCTGGCGTCTGTAC"; 
String codon = genome.substring(6, 

40 bytes 


3 ); 


object 

overhead 


16 


hash 


padding 


codon 


object 

overhead 


value 


hash 


padding 


40 bytes the string’s characters. When you use the substringO 

method, you create a new String object (40 bytes) 
but reuse the same val ue [] array, so a substring of an 
existing string takes just 40 bytes. The character array 
containing the original string is aliased in the object for 
the substring; the offset and length fields identify the 
substring. In other words, a substring takes constant ex¬ 
tra memory and forming a substring takes constant time, 
even when the lengths of the string and the substring 
are huge. A naive representation that requires copying 
characters to make substrings would take linear time 
and space. The ability to create a substring using space 
(and time) independent of its length is the key to effi¬ 
ciency in many basic string-processing algorithms. 

THESE BASIC MECHANISMS ARE EFFECTIVE for esti¬ 
mating the memory usage of a great many programs, 
but there are numerous complicating factors that can 
make the task significantly more difficult. We have 
already noted the potential effect of aliasing. More¬ 
over, memory consumption is a complicated dynamic 
process when function calls are involved because the 
system memory allocation mechanism plays a more 
important role, with more system dependencies. For 
example, when your program calls a method, the sys¬ 
tem allocates the memory needed for the method (for 
its local variables) from a special area of memory called 
the stack (a system pushdown stack), and when the 
method returns to the caller, the memory is returned 
to the stack. For this reason, creating arrays or other large objects in recursive programs 
is dangerous, since each recursive call implies significant memory usage. When you 
create an object with new, the system allocates the memory needed for the object from 
another special area of memory known as the heap (not the same as the binary heap 
data structure we consider in section 2 . 4 ), and you must remember that every object 
lives until no references to it remain, at which point a system process known as garbage 
collection reclaims its memory for the heap. Such dynamics can make the task of pre¬ 
cisely estimating memory usage of a program challenging. 


object 

overhead 

16 

C 

G 

C 

C 

T 

G 

G 

C 

G 

T 

C 

T 

G 

T 

A 

C 

padding 


x char 

.values 


36 bytes 


40 bytes 

A Stri ng and a substring 
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Perspective Good performance is important. An impossibly slow program is al¬ 
most as useless as an incorrect one, so it is certainly worthwhile to pay attention to the 
cost at the outset, to have some idea of which kinds of problems you might feasibly 
address. In particular, it is always wise to have some idea of which code constitutes the 
inner loop of your programs. 

Perhaps the most common mistake made in programming is to pay too much at¬ 
tention to performance characteristics. Your first priority is to make your code clear 
and correct. Modifying a program for the sole purpose of speeding it up is best left for 
experts. Indeed, doing so is often counterproductive, as it tends to create code that is 
complicated and difficult to understand. C. A. R. Hoare (the inventor of quicksort and 
a leading proponent of writing clear and correct code) once summarized this idea by 
saying that “ premature optimization is the root of all evil,” to which Knuth added the 
qualifier “(or at least most of it) in programming.” Beyond that, improving the running 
time is not worthwhile if the available cost benefits are insignificant. For example, im¬ 
proving the running time of a program by a factor of 10 is inconsequential if the run¬ 
ning time is only an instant. Even when a program takes a few minutes to run, the total 
time required to implement and debug an improved algorithm might be substantially 
more than the time required simply to run a slightly slower one—you may as well let 
the computer do the work. Worse, you might spend a considerable amount of time and 
effort implementing ideas that should in theory improve a program but do not do so 
in practice. 

Perhaps the second most common mistake made in programming is to ignore per¬ 
formance characteristics. Faster algorithms are often more complicated than brute- 
force ones, so you might be tempted to accept a slower algorithm to avoid having to 
deal with more complicated code. However, you can sometimes reap huge savings with 
just a few lines of good code. Users of a surprising number of computer systems lose 
substantial time unknowingly waiting for brute-force quadratic algorithms to finish 
solving a problem, when linear or linearithmic algorithms are available that could solve 
the problem in a fraction of the time. When we are dealing with huge problem sizes, we 
often have no choice but to seek better algorithms. 

We generally take as implicit the methodology described in this section to estimate 
memory usage and to develop an order-of-growth hypothesis of the running time from 
a tilde approximation resulting from a mathematical analysis within a cost model, and 
to check those hypotheses with experiments. Improving a program to make it more 
clear, efficient, and elegant should be your goal every time that you work on it. If you 
pay attention to the cost all the way through the development of a program, you will 
reap the benefits every time you use it. 
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| Q&A 

Q. Why not use StdRandom to generate random values instead of maintaining the file 
lMints.txt ? 

A. It is easier to debug code in development and to reproduce experiments. StdRandom 
produces different values each time it is called, so running a program after fixing a bug 
may not test the fix! You could use the initialize() method in StdRandom to address 
this problem, but a reference file such as IMi nts . txt makes it easier to add test cases 
while debugging. Also, different programmers can compare performance on different 
computers, without worrying about the input model. Once you have debugged a pro¬ 
gram and have a good idea of how it performs, it is certainly worthwhile to test it on 
random data. For example, Doubl i ngTest and Doubl i ngRatio take this approach. 

Q. I ran Doubl i ngRati o on my computer, but the results were not as consistent as in 
the book. Some of the ratios were not close to 8. Why? 

A. Thatiswhywediscussed“caveats”onpage 195. Most likely, your computer’s operating 
system decided to do something else during the experiment. One way to mitigate such 
problems is to invest more time in more experiments. For example, you could change 
Doubl i ngTest to run the experiments 1,000 times for each N, giving a much more ac¬ 
curate estimate for the running time for each size (see exercise 1 . 4 . 39 ). 

Q. What, exactly, does “as N grows” mean in the definition of the tilde notation? 

A. The formal definition of f(N) ~ g(N) is \im N ^ 00 f(N)/g(N) = 1. 

Q. I’ve seen other notations for describing order of growth. What’s the story? 

A. The “big-Oh” notation is widely used: we say that f(N) is ()(g( /V )) if there exist 
constants c and N 0 such that | f( N) | < c g(N) for all N > N 0 . This notation is very use¬ 
ful in providing asymptotic upper bounds on the performance of algorithms, which is 
important in the theory of algorithms. But it is not useful for predicting performance 
or for comparing algorithms. 

Q. Why not? 

A. The primary reason is that it describes only an upper bound on the running time. 
Actual performance might be much better. The running time of an algorithm might 
be both O ( N 2 ) and ~ a N log N. As a result, it cannot be used to justify tests like our 
doubling ratio test (see proposition c on page 193). 
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Q. So why is the big-Oh notation so widely used? 

A. It facilitates development of bounds on the order of growth, even for complicated 
algorithms for which more precise analysis might not be feasible. Moreover, it is com¬ 
patible with the “big-Omega” and “big-Theta” notations that theoretical computer sci¬ 
entists use to classify algorithms by bounding their worst-case performance. We say 
that/(iV) is fl ig(N)) if there exist constants c and N 0 such that |/(1V)| > cg{N) for N 
> N 0 ; and if f(N) is 0(g(N)) and £l(g(N)), we say that f(N) is ®(g(N)). The “big- 
Omega” notation is typically used to describe a lower bound on the worst case, and the 
“big-Theta” notation is typically used to describe the performance of algorithms that 
are optimal in the sense that no algorithm can have better asymptotic worst-case order 
of growth. Optimal algorithms are certainly worth considering in practical applica¬ 
tions, but there are many other considerations, as you will see. 

Q. Aren’t upper bounds on asymptotic performance important? 

A. Yes, but we prefer to discuss precise results in terms of frequency of statement ex¬ 
ecution with respect to cost models, because they provide more information about 
algorithm performance and because deriving such results is feasible for the algorithms 
that we discuss. For example, we say “ThreeSum uses ~N 3 1 2 array accesses” and “the 
number of times cnt++ is executed in ThreeSum is ~N 3 /6 in the worst case,” which is a 
bit more verbose but much more informative than the statement “the running time of 
ThreeSum is 0(iV 3 ).” 

Q. When the order of growth of the running time of an algorithm is N log N, the dou¬ 
bling test will lead to the hypothesis that the running time is ~ a N for a constant a. Isn’t 
that a problem? 

A. We have to be careful not to try to infer that the experimental data implies a par¬ 
ticular mathematical model, but when we are just predicting performance, this is not 
really a problem. For example, when N is between 16,000 and 32,000, the plots of 14N 
and N lg N are very close to one another. The data fits both curves. As N increases, the 
curves become closer together. It actually requires some care to experimentally check 
the hypothesis that an algorithm’s running time is linearithmic but not linear. 

Q. Does i nt [] a = new i nt [N] count as N array accesses (to initialize entries to 0)? 

A. Most likely yes, so we make that assumption in this book, though a sophisticated 
compiler implementation might try to avoid this cost for huge sparse arrays. 
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EXERCISES 


1 . 4.1 Show that the number of different triples that can be chosen from N items is 
precisely N(N— 1)(N—2)/6. Hint'. Use mathematical induction. 

1 . 4.2 Modify ThreeSum to work properly even when the i nt values are so large that 
adding two of them might cause overflow. 

1 . 4.3 Modify Doubl i ngTest to use StdDraw to produce plots like the standard and 
log-log plots in the text, rescaling as necessary so that the plot always fills a substantial 
portion of the window. 

1 . 4.4 Develop a table like the one on page 181 for TwoSum. 

1 . 4.5 Give tilde approximations for the following quantities: 

a. N + 1 

b. 1 + 1 IN 

c. (1 + 1/JV)(1 + 2 IN) 

d. 2,N 3 ~ 15 N 2 + N 

e. lg(2N)/lgN 

f. lg(iV 2 + 1) / lg AT 

g. N 100 / 2 N 

1 . 4.6 Give the order of growth (as a function of N) of the running times of each of the 
following code fragments: 

a. int sum = 0; 

for (int n = N; n > 0; n /= 2) 
for(int i =0; i < n; i++) 
sum++; 

b. int sum = 0; 

for (int i = 1 i < N; i *= 2) 
for (int j = 0; j < i; j++) 
sum++; 
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c. int sum = 0; 

for (int i = 1 i < N; i *= 2) 
for (int j =0; j < N; j++) 
sum++; 

1.4.7 Analyze ThreeSum under a cost model that counts arithmetic operations (and 
comparisons) involving the input numbers. 

1 .4.8 Write a program to determine the number pairs of values in an input file that 
are equal. If your first try is quadratic, think again and use Arrays. sort() to develop 
a linearithmic solution. 

1 .4.9 Give a formula to predict the running time of a program for a problem of size N 
when doubling experiments have shown that the doubling factor is 2 b and the running 
time for problems of size N 0 is T. 

1.4.10 Modify binary search so that it always returns the element with the smallest 
index that matches the search element (and still guarantees logarithmic running time). 

1.4.11 Add an instance method howManyO toStati cSEToflnts (page 99) that finds the 
number of occurrences of a given key in time proportional to log N in the worst case. 

1.4.12 Write a program that, given two sorted arrays of N i nt values, prints all ele¬ 
ments that appear in both arrays, in sorted order. The running time of your program 
should be proportional to N in the worst case. 

1.4.13 Using the assumptions developed in the text, give the amount of memory need¬ 
ed to represent an object of each of the following types: 

a. Accumulator 

b. Transaction 

c. Fi xedCapaci tyStackOfStri ngs with capacity C and N entries 

d. Point2D 

e. Interval ID 

f. Interval2D 

g. Double 
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| CREATIVE PROBLEMS 

1.4.14 4-sum. Develop an algorithm for the 4-sum problem. 

1.4.15 Faster 3-sum. As a warmup, develop an implementation TwoSumFaster that 
uses a linear algorithm to count the pairs that sum to zero after the array is sorted (in¬ 
stead of the binary-search-based linearithmic algorithm). Then apply a similar idea to 
develop a quadratic algorithm for the 3-sum problem. 

1.4.16 Closest pair (in one dimension). Write a program that, given an array a [] of N 
double values, finds a closest pair: two values whose difference is no greater than the 
the difference of any other pair (in absolute value). The running time of your program 
should be linearithmic in the worst case. 

1.4.17 Farthest pair (in one dimension). Write a program that, given an array a [] of N 
doubl e values, finds a farthest pair: two values whose difference is no smaller than the 
the difference of any other pair (in absolute value). The running time of your program 
should be linear in the worst case. 

1.4.18 Local minimum of an array. Write a program that, given an array a [] of N dis¬ 
tinct integers, finds a local minimum: an index i such that a[i -1] < a[i] < a[i+l]. 
Your program should use ~2lg N compares in the worst case.. 

Answer: Examine the middle value a[N/2] and its two neighbors a[N/2 - 1] and 
a[N/2 + 1]. If a[N/2] is a local minimum, stop; otherwise search in the half with the 
smaller neighbor. 

1.4.19 Local minimum of a matrix. Given an N-by-N array a[] of N 2 distinct inte¬ 
gers, design an algorithm that runs in time proportional to N to find a local minimum: 
a pair of indices i and j such that a[i ] [j] < a[i+l] [j], a[i] [j] < a[i][j+l], 
a[i][j] < a[i-1 ] [j], and a[i ] [j] < a[i ] [j-1]. The running time of your pro¬ 
gram should be proportional to N in the worst case. 

1.4.20 Bitonic search. An array is bitonic if it is comprised of an increasing sequence 
of integers followed immediately by a decreasing sequence of integers. Write a program 
that, given a bitonic array of N distinct i nt values, determines whether a given integer 
is in the array. Your program should use ~3lg N compares in the worst case. 

1.4.21 Binary search on distinct values. Develop an implementation of binary search 
for Stati cSEToflnts (see page 98) where the running time of contai ns() is guaranteed 
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to be ~ lg R, where R is the number of different integers in the array given as argument 
to the constructor. 

1 . 4.22 Binary search with only addition and subtraction. [Mihai Patrascu] Write a 
program that, given an array of N distinct i nt values in ascending order, determines 
whether a given integer is in the array You may use only additions and subtractions 
and a constant amount of extra memory. The running time of your program should be 
proportional to log N in the worst case. 

Answer-. Instead of searching based on powers of two (binary search), use Fibonacci 
numbers (which also grow exponentially). Maintain the current search range to be the 
interval [i, i + F k \ and keep F k and F k] in two variables. At each step compute F k _ 2 via 
subtraction, check element i + F k _ 2 , and update the current range to either [i, i + F k _ 2 ] 
or [i + F k _ 2 ,i + F k _ 2 + F k _ l \. 

1 . 4.23 Binary search for a fraction. Devise a method that uses a logarithmic number of 
queries of the form Is the number less than x? to find a rational number p/q such that 0 
< p < q < N. Hint: Two fractions with denominators less than N cannot differ by more 
than 1 /N 2 . 

1 . 4.24 Throwing eggs from a building. Suppose that you have an N- story building and 
plenty of eggs. Suppose also that an egg is broken if it is thrown off floor F or higher, 
and unhurt otherwise. First, devise a strategy to determine the value of F such that the 
number of broken eggs is ~lg N when using ~lg N throws, then find a way to reduce the 
cost to ~2lg F. 

1 . 4.25 Throwing two eggs from a building. Consider the previous question, but now 
suppose you only have two eggs, and your cost model is the number of throws. Devise 
a strategy to determine F such that the number of throws is at most 2 \N, then find a 
way to reduce the cost to ~cVF. This is analogous to a situation where search hits (egg 
intact) are much cheaper than misses (egg broken). 

1 . 4.26 3-collinearity. Suppose that you have an algorithm that takes as input N dis¬ 
tinct points in the plane and can return the number of triples that fall on the same line. 
Show that you can use this algorithm to solve the 3-sum problem. Strong hint: Use 
algebra to show that (a, a 3 ), ( b , b 3 ), and (c, c 3 ) are collinear if and only if a + b + c = 0. 

1 . 4.27 Queue with two stacks. Implement a queue with two stacks so that each queue 
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CREATIVE PROBLEMS (continued) 


operation takes a constant amortized number of stack operations. Hint: If you push 
elements onto a stack and then pop them all, they appear in reverse order. If you repeat 
this process, they’re now back in order. 

1.4.28 Stack with a queue. Implement a stack with a single queue so that each stack 
operations takes a linear number of queue operations. Hint: To delete an item, get all 
of the elements on the queue one at a time, and put them at the end, except for the last 
one which you should delete and return. (This solution is admittedly very inefficient.) 

1.4.29 Steque with two stacks. Implement a steque with two stacks so that each steque 
operation (see exercise 1 . 3 . 32 ) takes a constant amortized number of stack operations. 

1.4.30 Deque with a stack and a steque. Implement a deque with a stack and a steque 
(see exercise 1 . 3 . 32 ) so that each deque operation takes a constant amortized number 
of stack and steque operations. 

1.4.31 Deque with three stacks. Implement a deque with three stacks so that each 
deque operation takes a constant amortized number of stack operations. 

1.4.32 Amortized analysis. Prove that, starting from an empty stack, the number of ar¬ 
ray accesses used by any sequence of M operations in the resizing array implementation 
of Stack is proportional to M. 

1.4.33 Memory requirements on a 32-bit machine. Give the memory requirements 
for Integer, Date, Counter, int[], doublet], doublet] □ > String, Node, and Stack 
(linked-list representation) for a 32-bit machine. Assume that references are 4 bytes, 
object overhead is 8 bytes, and padding is to a multiple of 4 bytes. 

1.4.34 Hot or cold. Your goal is to guess a secret integer between 1 and N. You repeat¬ 
edly guess integers between 1 and N. After each guess you learn if your guess equals the 
secret integer (and the game stops). Otherwise, you learn if the guess is hotter (closer to) 
or colder (farther from) the secret number than your previous guess. Design an algo¬ 
rithm that finds the secret number in at most ~2 lg N guesses. Then design an algorithm 
that finds the secret number in at most ~ 1 lg N guesses. 
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1.4.35 Time costs for pushdown stacks. Justify the entries in the table below, which 
shows typical time costs for various pushdown stack implementations, using a cost 
model that counts both data references (references to data pushed onto the stack, either 
an array reference or a reference to an object’s instance variable) and objects created. 


data structure 


item type 


cost to push \ i nt values 
data references objects created 



int 

IN 

N 

linked list 


Integer 

3 N 

2 N 


int 

~5 N 

lg N 

resizing array 


Integer 

~5N 

~N 


Time costs for pushdown stacks (various implementations) 


1.4.36 Space usage for pushdown stacks. Justify the entries in the table below, which 
shows typical space usage for various pushdown stack implementations. Use a static 
nested class for linked-list nodes to avoid the non-static nested class overhead. 


data structure item type space usage for AM nt values (bytes) 


linked list 


int 


Integer 


-32 N 


- 64 AT 


resizing array 


between 
~4 N and ~1 6N 

between 

~32 N and ~56 N 

Space usage in pushdown stacks (various implementations) 


int 


Integer 
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| EXPERIMENTS 


1 . 4.37 Autoboxing performance penalty. Run experiments to determine the perfor¬ 
mance penalty on your machine for using autoboxing and auto-unboxing. Develop an 
implementation Fi xedCapaci tyStackOf Ints and use a client such as Doubl i ngRati o 
to compare its performance with the generic Fi xedCapaci tyStack<Integer>, for a 
large number of push() and popO operations. 

1 . 4.38 Naive 3-sum implementation. Run experiments to evaluate the following im¬ 
plementation of the inner loop of ThreeSum: 

for (int i =0; i < N; i++) 
for (int j =0; j < N; j++) 
for (int k = 0; k < N; k++) 
if (i < j && j < k) 

if Ca[i] + a[j] + a[k] == 0) 
cnt++; 

Do so by developing a version of Doubl i ngTest that computes the ratio of the running 
times of this program and ThreeSum. 

1 . 4.39 Improved accuracy for doubling test. Modify Doubl i ngRati o to take a second 
command-line argument that specifies the number of calls to make to ti meTri al () for 
each value of N. Run your program for 10, 100, and 1,000 trials and comment on the 
precision of the results. 

1 . 4.40 3-sum for random values. Formulate and validate a hypothesis describing the 
number of triples of N random i nt values that sum to 0. If you are skilled in math¬ 
ematical analysis, develop an appropriate mathematical model for this problem, where 
the values are uniformly distributed between -M and M, where M is not small. 

1 . 4.41 Running times. Estimate the amount of time it would take to run TwoSumFast, 
TwoSum, Th reeSumFast and Th reeSum on your computer to solve the problems for a file 
of 1 million numbers. Use Doubl i ngRati o to do so. 

1 . 4.42 Problem sizes. Estimate the size of the largest value of P for which you can run 
TwoSumFast, TwoSum, Th reeSumFast, and ThreeSum on your computer to solve the 
problems for a file of 2 P thousand numbers. Use Doubl i ngRati o to do so. 

1 . 4.43 Resizing arrays versus linked lists. Run experiments to validate the hypothesis 
that resizing arrays are faster than linked lists for stacks (see exercise 1 . 4.35 and exer¬ 
cise 1 . 4 . 36 ). Do so by developing a version of Doubl i ngRati o that computes the ratio 
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of the running times of the two programs. 

1 . 4.44 Birthday problem. Write a program that takes an integer N from the command 
line and uses StdRandom. uni form() to generate a random sequence of integers be¬ 
tween 0 and N — 1. Run experiments to validate the hypothesis that the number of 
integers generated before the first repeated value is found is ~Vtt ~N/2. 

1 . 4.45 Coupon collector problem. Generating random integers as in the previous exer¬ 
cise, run experiments to validate the hypothesis that the number of integers generated 
before all possible values are generated is ~NH n . 
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1.5 CASE STUDY: UNION-FIND 



to illustrate our basic approach to developing and analyzing algorithms, we now 
consider a detailed example. Our purpose is to emphasize the following themes. 

■ Good algorithms can make the difference between being able to solve a practical 
problem and not being able to address it at all. 

■ An efficient algorithm can be as simple to code as an inefficient one. 

■ Understanding the performance characteristics of an implementation can be an 
interesting and satisfying intellectual challenge. 

■ The scientific method is an important tool in helping us choose among different 
methods for solving the same problem. 

■ An iterative refinement process can lead to increasingly efficient algorithms. 
These themes are reinforced throughout the book. This prototypical example sets the 
stage for our use of the same general methodology for many other problems. 

The problem that we consider is not a toy problem; it is a fundamental compu¬ 
tational task, and the solution that we develop is of use in a variety of applications, 
from percolation in physical chemistry to connectivity in communications networks. 
We start with a simple solution, then seek to understand that solution’s performance 
characteristics, which help us to see how to improve the algorithm. 

Dynamic connectivity We start with the following problem specification: The 
input is a sequence of pairs of integers, where each integer represents an object of some 
type and we are to interpret the pair p q as meaning “p is connected to q.” We assume 
that “is connected to” is an equivalence relation, which means that it is 

■ Reflexive: p is connected to p. 

■ Symmetric: If p is connected to q, then q is connected to p. 

■ Transitive: If p is connected to q and q is connected to r, then p is connected to r. 
An equivalence relation partitions the objects into equivalence classes. In this case, two 
objects are in the same equivalence class if and only if they are connected. Our goal is 
to write a program to filter out extraneous pairs (pairs where both objects are in the 
same equivalence class) from the sequence. In other words, when the program reads a 
pair p q from the input, it should write the pair to the output only if the pairs it has 
seen to that point do not imply that p is connected to q. If the previous pairs do imply 
that p is connected to q, then the program should ignore the pair p q and proceed to 
read in the next pair. The figure on the facing page gives an example of this process. To 
achieve the desired goal, we need to devise a data structure that can remember sufficient 
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information about the pairs it has seen to be able to decide whether or not a new pair of 
objects is connected. Informally, we refer to the task of designing such a method as the 
dynamic connectivity problem. This problem arises applications such as the following: 


Networks. The integers might represent computers in a large network, and the pairs 
might represent connections in the network. Then, our program determines whether 
we need to establish a new direct connection for p and q to be able 
to communicate or whether we can use existing connections to 
set up a communications path. Or, the integers might represent 
contact sites in an electrical circuit, and the pairs might represent 
wires connecting the sites. Or, the integers might represent people 4 
in a social network, and the pairs might represent friendships. In 
such applications, we might need to process millions of objects 
and billions of connections. 


3 8 


6 5 


0 1 

• 

5 6 

2 3 4 

• • • 

7 8 9 

• 

. .—. 

• 

• 

• r 

• 

• r 


9 4 


2 1 


n 

n 


Variable-name equivalence. In certain programming environ¬ 
ments, it is possible to declare two variable names as being equiv¬ 
alent (references to the same object). After a sequence of such dec¬ 
larations, the system needs to be able to determine whether two 
given names are equivalent. This application is an early one (for 
the FORTRAN programming language) that motivated the devel¬ 
opment of the algorithms that we are about to consider. 

Mathematical sets. On a more abstract level, you can think of 
the integers as belonging to mathematical sets. When we process a 
pair p q, we are asking whether they belong to the same set. If not, 
we unite p’s set and q’s set, putting them in the same set. 

to fix ideas, we will use networking terminology for the rest of 
this section and refer to the objects as sites, the pairs as connec¬ 
tions, and the equivalence classes as connected components, or just 
components for short. For simplicity, we assume that we have N 
sites with integer names, from 0 to N-l. We do so without loss of 
generality because we shall be considering a host of algorithms in 
chapter 3 that can associate arbitrary names with such integer 
identifiers in an efficient manner. 

A larger example that gives some indication of the difficulty of the connectivity 
problem is depicted in the figure at the top of the next page. You can quickly identify 
the component consisting of a single site in the left middle of the diagram and the 



don t print 
pairs that 
are already 
connected 


Dynamic connectivity example 
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connected 

component 



Medium connectivity example (625 sites, 900 edges, 3 connected components) 


component consisting of five sites at the bottom left, but you might have difficulty veri¬ 
fying that all of the other sites are connected to one another. For a program, the task is 
even more difficult, because it has to work just with site names and connections and has 
no access to the geometric placement of sites in the diagram. How can we tell quickly 
whether or not any given two sites in such a network are connected? 

The first task that we face in developing an algorithm is to specify the problem in a 
precise manner. The more we require of an algorithm, the more time and space we may 
expect it to need to finish the job. It is impossible to quantify this relationship a priori, 
and we often modify a problem specification on finding that it is difficult or expensive 
to solve or, in happy circumstances, on finding that an algorithm can provide informa¬ 
tion more useful than what was called for in the original specification. For example, our 
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connectivity problem specification requires only that our program be able to determine 
whether or not any given pair p q is connected, and not that it be able to demonstrate a 
set of connections that connect that pair. Such a requirement makes the problem more 
difficult and leads us to a different family of algorithms, which we consider in section 
4 . 1 . 

To specify the problem, we develop an API that encapsulates the basic operations 
that we need: initialize, add a connection between two sites, identify the component 
containing a site, determine whether two sites are in the same component, and count 
the number of components. Thus, we articulate the following API: 


public class UF 



UFCint N) 



voi d 

uni on(int p, 

int 

q) 

int 

find (int p) 



boolean 

connected(int 

P> 

int 

int 

countO 




initialize N sites with integer names (0 to N-l) 
add connection between p and q 
component identifier for p (0 to N-l) 
return true if p and q are in the same compom 
number of components 
Union-find API 


The uni on () operation merges two components if the two sites are in different com¬ 
ponents, the fi nd() operation returns an integer component identifier for a given site, 
the connected O operation determines whether two sites are in the same component, 
and the countO method returns the number of components. We start with N compo¬ 
nents, and each uni on() that merges two different components decrements the num¬ 
ber of components by 1. 

As we shall soon see, the development of an algorithmic solution for dynamic con¬ 
nectivity thus reduces to the task of developing an implementation of this API. Every 
implementation has to 

■ Define a data structure to represent the known connections 

■ Develop efficient uni on (), fi nd(), connectedO, and countO implementa¬ 
tions that are based on that data structure 

As usual, the nature of the data structure has a direct impact on the efficiency of the 
algorithms, so data structure and algorithm design go hand in hand. The API already 
specifies the convention that both sites and components will be identified by i nt val¬ 
ues between 0 and N-l, so it makes sense to use a site-indexed array i d [] as our basic 
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data structure to represent the components. We always use the name of one of the sites 
in a component as the component identifier, so you can think of each component as 
being represented by one of its sites. Initially, we start with N components, each site in 
its own component, so we initialize id[i] to i for all i from 0 to N-l. For each site 
i, we keep the information needed by fi nd() to determine the component contain¬ 
ing i in i d [i ], using various algorithm-dependent strategies. All of our implementa¬ 
tions use a one-line implementation of connectedO that returns the boolean value 
find(p) == find(q). 

in summary, our starting point is algorithm 1.5 on the facing 
page. We maintain two instance variables, the count of components 
and the array id[] . Implementations of find() and union() are 
the topic of the remainder of this section. 

To test the utility of the API and to provide a basis for develop¬ 
ment, we include a client in main() that uses it to solve the dy¬ 
namic connectivity problem. It reads the value of N followed by a 
sequence of pairs of integers (each in the range 0 to N-l), calling 
fi nd() for each pair: If the two sites in the pair are already con¬ 
nected, it moves on to the next pair; if they are not, it calls uni on () 
and prints the pair. Before considering implementations, we also 
prepare test data: the file tinyUF.txt contains the 11 connections 
among 10 sites used in the small example illustrated on page 217, the 
file mediumUF.txt contains the 900 connections among 625 sites 
illustrated on page 218, and the file IargeUF.txt is an example with 
2 million connections among 1 millions sites. Our goal is to be able 
to handle inputs such as IargeUF.txt in a reasonable amount of 
time. 

To analyze the algorithms, we focus on the number of times each 
algorithm accesses an array entry. By doing so, we are implicitly for¬ 
mulating the hypothesis that the running times of the algorithms 
on a particular machine are 
within a constant factor of 
this quantity. This hypothesis is immediate from 
the code, is not difficult to validate through ex¬ 
perimentation, and provides a useful starting 
point for comparing algorithms, as we will see. 


Union-find cost model. When 
studying algorithms to imple¬ 
ment the union-find API, we 
count array accesses (the num¬ 
ber of times an array entry is 
accessed, for read or write). 


% more tinyUF.txt 
10 

4 3 
3 8 

6 5 
9 4 
2 1 
8 9 

5 0 

7 2 

6 1 
1 0 
6 7 

% more mediumUF.txt 
625 

528 503 
548 523 

[900 connections] 

% more IargeUF.txt 
1000000 
786321 134521 
696834 98245 

[2000000 connections] 
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ALGORITHM 1.5 Union-find implementation 


public class UF 

{ 

private int[] id; // access to component id (site indexed) 

private int count; // number of components 


public UF(int N) 

{ // Initialize component id array. 

count = N; 
id = new int[N]; 
for (int i =0; i < N; i++) 
i d [i ] = i ; 

} 

public int countO 
{ return count; } 

public boolean connected(int p, int q) 
{ return find(p) == find(q); } 


% java UF < tinyUF.txt 

4 B 
B 8 

6 5 
9 4 
2 1 

5 0 

7 2 

6 1 

2 components 


public int find(int p) 
public void union(int p, int q) 

// See page 222 (quick-find),page 224 (quick-union) andpage 228 (weighted). 


public static void main(String[] args) 

{ // Solve dynamic connectivity problem on Stdln. 

int N = Stdln.readlnt(); // Read number of sites. 

UF uf = new UF(N); // Initialize N components, 

while (!Stdln.isEmptyO) 

{ 


int p = Stdln.readlnt(); 
int q = Stdln.readlnt(); 
if (uf.connected(p, q)) continue; 
uf.union(p, q); 

StdOut.println(p + " " + q); 


// Read pair to connect. 

// Ignore if connected. 

// Combine components 
// and print connection. 


} 

StdOut.println(uf.count() + " components"); 


} 


} 


Our UF implementations are based on this code, which maintains an array of integers id[] such 
that the fi nd() method returns the same integer for every site in each connected component. The 
uni on () method must maintain this invariant. 
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Implementations We shall consider three different implementations, all based on 
using the site-indexed i d [] array, to determine whether two sites are in the same con¬ 
nected component. 

Quick-find. One approach is to maintain the invariant that p and q are connected 
if and only if i d [p] is equal to i d [q] . In other words, all sites in a component must 
have the same value in id[]. This method is called quick-find because find(p) just 
returns i d [p] , which immediately implies that connected (p, q) reduces to just the 
test id[p] == id[q] and returns true if and only 
if p and q are in the same component. To maintain 
the invariant for the call uni on (p, q), we first check 
whether they are already in the same component, in 
which case there is nothing to do. Otherwise, we are 
faced with the situation that all of the id[] entries 
corresponding to sites in the same component as 
p have one value and all of the id[] entries corre¬ 
sponding to sites in the same component as q have 
another value. To combine the two components into 
one, we have to make all of the id[] entries cor¬ 
responding to both sets of sites the same value, as 
shown in the example at right. To do so, we go through the array, changing all the entries 
with values equal to i d [p] to the value i d [q] . We could have decided to change all the 
entries equal to i d [q] to the value i d [p] —the choice between these two alternatives 

is arbitrary. The code for fi nd() and 
unionO based on these descriptions, 
given at left, is straightforward. A full 
trace for our development client with 
our sample test data tinyUF.txt is 
shown on the next page. 


public int find(int p) 

{ return id[p] ; } 

public void uni on (int p, int q) 

{ // Put p and q into the same component. 

i nt pID = fi nd(p) ; 
int qlD = fi nd(q); 

// Nothing to do if p and q are already 
in the same component, 
if (pID == qlD) return; 

// Rename p’s component to q’s name. 

for (int i =0; i < id.length; i++) 
if (id[i] == pID) id[i] = qlD; 
count—; 


Quick-find 


find examines i d [5] and i d [9] 

pq 012B456789 

59 1118811188 

union has to change all Is to 8s 

pq 012B456789 

59 1118811188 

888888888 

Quick-find overview 
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Quick-find analysis. The fi nd() operation is certainly quick, as it only accesses the 
i d [] array once in order to complete the operation. But quick-find is typically not use¬ 
ful for large problems because uni on () needs to scan through the whole i d [] array for 
each input pair. 

Proposition F. The quick-find algorithm uses one array access for each call to 
fi nd() and between N + 3 and 2N+ 1 array accesses for each call to uni on 0 that 
combines two components. 

Proof: Immediate from the code. Each call to connected () tests two entries in the 
i d [] array, one for each of the two calls to f i nd (). Each call to uni on () that com¬ 
bines two components does so by making two calls to f i nd (), testing each of the N 
entries in the i d [] array, and changing between 1 and N— 1 of them. 


id[] 


p 

q 

0 

1 

2 

3 

4 

5 

6 

7 

00 

9 

4 

3 

0 

1 

2 

3 

4 

5 

6 

7 

8 

9 



0 

1 

2 

3 

3 

5 

6 

7 

8 

9 

3 

8 

0 

1 

2 

3 

3 

5 

6 

7 

8 

9 



0 

1 

2 

8 

8 

5 

6 

7 

8 

9 

6 

5 

0 

1 

2 

8 

8 

5 

6 

7 

8 

9 



0 

1 

2 

8 

8 

5 

5 

7 

8 

9 

9 

4 

0 

1 

2 

8 

8 

5 

5 

7 

8 

9 



0 

1 

2 

8 

8 

5 

5 

7 

8 

8 

2 

1 

0 

1 

2 

8 

8 

5 

5 

7 

8 

8 



0 

1 

1 

8 

8 

5 

5 

7 

8 

8 

00 

9 

0 

1 

1 

8 

8 

5 

5 

7 

8 

8 

5 

0 

0 

1 

1 

8 

8 

5 

5 

7 

8 

8 



0 

1 

1 

8 

8 

0 

0 

7 

8 

8 

7 

2 

0 

1 

1 

8 

8 

0 

0 

7 

8 

8 



0 

1 

1 

8 

8 

0 

0 

1 

8 

8 

6 

1 

0 

1 

1 

8 

8 

0 

0 

1 

8 

8 



1 

1 


8 

8 

1 

fl 

1 

8 

8 

1 

0 

1 

1 

1 

Ss 

V 8 


ll 

1 

8 

8 

6 

7 



1 

8 



1 

1 

8 

8 


\ id[p] and i d [q] differ, so 
\ uni on () changes entries equal 
\ toid[p] foid[q] (in red) 
id[p] and id[q] 
match, so no change 

Quick-find trace 


In particular, suppose that we use quick-find for the 
dynamic connectivity problem and wind up with a 
single component. This requires at least N— 1 calls to 
uni on (), and, consequently, at least {N+3){N— 1) ~ 
N 2 array accesses—we are led immediately to the hy¬ 
pothesis that dynamic connectivity with quick-find 
can be a quadratic- time process. This analysis gener¬ 
alizes to say that quick-find is quadratic for typical 
applications where we end up with a small number of 
components. You can easily validate this hypothesis 
on your computer with a doubling test (see exercise 
1 . 5.23 for an instructive example). Modern comput¬ 
ers can execute hundreds of millions or billions of in¬ 
structions per second, so this cost is not noticeable if 
N is small, but we also might find ourselves with mil¬ 
lions or billions of sites and connections to process in 
a modern application, as represented by our test file 
largeUF.txt. If you are still not convinced and feel 
that you have a particularly fast computer, try using 
quick-find to determine the number of components 
implied by the pairs in largeUF.txt. The inescap¬ 
able conclusion is that we cannot feasibly solve such 
a problem using the quick-find algorithm, so we seek 
better algorithms. 
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private int find(int p) 

{ // Find component name. 

while (p != id[p]) p = id[p] ; 
return p; 

} 

public void uni on (int p, int q) 

{ // Give p and q the same root. 

int pRoot = find(p); 
int qRoot = find(q); 
if (pRoot == qRoot) return; 


Quick-union. The next algorithm that we consider is a complementary method that 
concentrates on speeding up the union() operation. It is based on the same data 
structure—the site-indexed id[] ar¬ 
ray—but we interpret the values dif¬ 
ferently, to define more complicated 
structures. Specifically, the i d [] entry 
for each site is the name of another 
site in the same component (possibly 
itself)—we refer to this connection as 
a link. To implement fi ndO, we start 
at the given site, follow its link to an¬ 
other site, follow that site’s link to yet 
another site, and so forth, following 
links until reaching a root, a site that 
has a link to itself (which is guaran¬ 
teed to happen, as you will see). Two 
sites are in the same component if and 
only if this process leads them to the 

same root. To validate this process, we need union(p, q)to maintain this invariant, 
which is easily arranged: we follow links to find the roots associated with p and q, then 
rename one of the components by linking one of these roots to the other; hence the 
name quick-union. Again, we have an arbitrary choice of whether to rename the com¬ 
ponent containing p or the component containing q; the implementation above re¬ 
names the one containing p. The 


} 


id[pRoot] 
count—; 


= qRoot; 


Quick-union 


id[] is parent-link representation 
of a forest of trees 



find has to follow links to the root 

pq 0 1 2 3 4 5 6 7 8 9 


5 9 1 1 1 8 3 0 5 


find(5) is 
i d [id [i d [5] ] ] 


8 8 

t 

find(9) is 
id[id[9]] 


figure on the next page shows a 
trace of the quick-union algo¬ 
rithm for ti nyUF. txt. This trace 
is best understood in terms of the 
graphical representation depict¬ 
ed at left, which we consider next. 


union changes just one link 

pq 0123456789 


5 9 


111 

18 1 


18 8 

5 18 8 
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Forest-of-trees representation. The code for quick-union is compact, but a bit opaque. 
Representing sites as nodes (labeled circles) and links as arrows from one node to an¬ 
other gives a graphical representation of the data structure that makes it relatively easy 
to understand the operation of the algorithm. The resulting structures are trees —in 
technical terms, our id[] array 


p q 


id[] 

0123456789 


4 3 


3 8 


6 5 


9 4 


2 1 


8 9 

5 0 


7 2 


6 1 


1 0 
6 7 


0 12 3 
0 12 3 
0 12 3 
0 12 8 


1 2 
1 2 

1 2 

1 2 


0 12 

0 11 

0 11 
0 1 
0 11 


1 1 

1 1 


8 3 0 
8 3 0 


0 11 

111 

111 

111 


9 

9 

9 

9 

9 

9 

8 9 
8 

8 8 

8 8 

8 8 

8 8 
8 8 

8 8 
8 8 

8 8 
8 8 

8 8 
8 8 


2 )©©®©©©®®© 

®®©©©©®®® 


® © © © © © 


© © © 


© 



is a parent-link representation 
of a forest (set) of trees. To sim¬ 
plify the diagrams, we often omit 
both the arrowheads in the links 
(because they all point upwards) 
and the self-links in the roots 
of the trees. The forests corre¬ 
sponding to the id[] array for 
tinytlF.txt are shown at right. 

When we start at the node cor¬ 
responding to any site and follow 
links, we eventually end up at the 
root of the tree containing that 
node. We can prove this prop¬ 
erty to be true by induction: It is 
true after the array is initialized 
to have every node link to itself, 
and if it is true before a given 
uni on() operation, it is certainly 
true afterward. Thus, the fi nd() 
method on page 224 returns the 
name of the site at the root (so 
that connected O checks wheth¬ 
er two sites are in the same tree). 

This representation is useful for 
this problem because the nodes 
corresponding to two sites are in 
the same tree if and only if the 
sites are in the same component. 

Moreover, the trees are not difficult to build: the uni on () implementation on page 224 
combines two trees into one in a single statement, by making the root of one the parent 
of the other. 





Quick-union trace (with corresponding forests of trees) 
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Quick-union analysis. The quick-union algorithm would seem to be faster than the 
quick-find algorithm, because it does not have to go through the entire array for each 

input pair; but how much faster is it? Analyzing the 
cost of quick-union is more difficult than it was for 
quick-find, because the cost is more dependent on 
the nature of the input. In the best case, f i nd () just 
needs one array access to find the identifier associ¬ 
ated with a site, as in quick-find; in the worst case, it 
needs 2 N + 1 array accesses, as for 0 in the example 
at left (this count is conservative since compiled 
code will typically not do an array access for the 
second reference to i d [p] in the whi 1 e loop). Ac¬ 
cordingly, it is not difficult to construct a best-case 
input for which the running time of our dynamic 
connectivity client is linear; on the other hand it is 
also not difficult to construct a worst-case input for 
which the running time is quadratic (see the dia¬ 
gram at left and proposition g below). Fortunate¬ 
ly, we do not need to face the problem of analyzing 
quick union and we will not dwell on comparative 
performance of quick-find and quick-union be¬ 
cause we will next examine another variant that is far more efficient than either. For the 
moment, you can regard quick-union as an improvement over quick-find because it 
removes quick-find’s main liability (that uni on () always takes linear time). This differ¬ 
ence certainly represents an improvement for typical data, but quick-union still has the 
liability that we cannot guarantee it to be substantially faster than quick-find in every 
case (for certain input data, quick-union is no faster than quick-find). 

Definition. The size of a tree is its number of nodes. The depth of a node in a tree 
is the number of links on the path from it to the root. The height of a tree is the 
maximum depth among its nodes. 


id[] 


p q 


o 1 


o 2 


0 B 


0 4 


0 12 3 4 


0 12 

112 3 4 

0 12 3 

1 2 2 3 4 

0 12 3 4 
1 2 3 3 4 


0 12 3 4 

1 2 3 4 4 



depth 4 - 

Quick-union worst case 


Proposition G. The number of array accesses used by fi nd() in quick-union is 1 
plus the twice the depth of the node corresponding to the given site. The number 
of array accesses used by uni on() and connected () is the cost of the two fi nd() 
operations (plus 1 for uni on() if the given sites are in different trees). 

Proof: Immediate from the code. 
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Again, suppose that we use quick-union for the dynamic connectivity problem and 
wind up with a single component. An immediate implication of proposition g is that 
the running time is quadratic, in the worst case. Suppose that the input pairs come 
in the order 0-1, then 0-2, then 0-3, and so forth. After N — 1 such pairs, we have N 
sites all in the same set, and the tree that is formed by the quick-union algorithm has 
height N— 1, with 0 linking to 1, which links to 2, which links to 3, and so forth (see 
the diagram on the facing page). By proposition g, the number of array accesses for 
the uni on () operation for the pair 0 i is exactly 2 i + 2 (site 0 is at depth i and site i at 
depth 0). Thus, the total number of array accesses for the fi nd() operations for these 
N pairs is 2 (1 + 2 + ... + N) ~N 2 . 

Weighted quick-union. Fortunately, there is an 
easy modification to quick-union that allows us 
to guarantee that bad cases such as this one do 
not occur. Rather than arbitrarily connecting the 
second tree to the first for uni on(), we keep track 
of the size of each tree and always connect the 
smaller tree to the larger. This change requires 
slightly more code and another array to hold the 
node counts, as shown on page 228, but it leads to 
substantial improvements in efficiency. We refer 
to this algorithm as the weighted quick-union al¬ 
gorithm. The forest of trees constructed by this 
algorithm for ti nyUF. txt is shown in the figure 

at left on the top of page 229. Even for this small example, the tree height is substantially 
smaller than the height for the unweighted version. 



weighted 



always chooses the 
better alternative 


Weighted quick-union 



Weighted quick-union analysis. The figure at right on the top of page 229 illustrates 
the worst case for weighted quick union, when the sizes of the trees to be merged by 


uni on () are always equal (and a power 
of 2). These tree structures look complex, 
but they have the simple property that 
the height of a tree of 2" nodes is n. Fur¬ 
thermore, when we merge two trees of 2” 
nodes, we get a tree of 2' 1+1 nodes, and we 
increase the height of the tree to n+\. This 
observation generalizes to provide a proof 
that the weighted algorithm can guarantee 
logarithmic performance. 


% java WeightedQuickUnionUF < mediumUF.txt 
528 503 
548 523 

3 components 

% java WeightedQuickUnionUF < largeUF.txt 
786321 134521 
696834 98245 

6 components 
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ALGORITHM 1.5 (continued) Union-find implementation (weighted quick-union) 


public class WeightedQuickUnionUF 

{ 

private int[] id; // parent link (site indexed) 

private int[] sz; // size of component for roots (site indexed) 

private int count; // number of components 

public WeightedQuicl<UnionUF(int N) 

{ 


count = N; 




id = 

new 

int[N]; 




for 

(int 

i = 0; i 

< N; 

i++) 

id[i] = 

sz = 

new 

int[N]; 




for 

(int 

i = 0; i 

< N; 

i++) 

sz[i] = 


} 

public int countO 
{ return count; } 

public boolean connected(int p, int q) 
{ return find(p) == find(q); } 


private int find(int p) 

{ // Follow links to find a root, 

while (p != id[p]) p = id[p]; 
return p; 

} 

public void union(int p, int q) 

{ 

int i = find(p); 
int j = find(q); 
if (i == j) return; 

// Make smaller root point to larger one. 
if Csz[i] < sz[j]) { id[i] = j; sz[j] += sz[i]; } 

else { id[j] = i; sz[i] += sz[j]; } 

count--; 

} 

} 


This code is best understood in terms of the forest-of-trees representation described in the text. We 
add a site-indexed array sz [] as an instance variable so that union () can link the root of the smaller 
tree to the root of the larger tree. This addition makes it feasible to address large problems. 
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reference input 

p q 
4 3 


worst-case input 


3 8 

6 5 

9 4 

2 1 

8 9 

5 0 

7 2 

6 1 
1 0 

6 7 


© © ® © © © ® 

© © © © © ® ® 
© 


p q 
o 1 
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Proposition H. The depth of any node in a forest built by weighted quick-union for 
N sites is at most lg N. 

Proof: We prove a stronger fact by (strong) induction: The height of every tree of 
size k in the forest is at most lg k. The base case follows from the fact that the tree 
height is 0 when k is 1. By the inductive hypothesis, assume that the tree height of a 
tree of size i is at most lg i for all i < k. When we combine a tree of size i with a tree 
of size j with i < j and i+j= k, we increase the depth of each node in the smaller set 
by 1, but they are now in a tree of size i+j= k, so the property is preserved because 
1 + lg i = lg(z + z) - lg(z +j)=lgk. 
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Corollary. For weighted quick-union with N sites, the worst-case order of growth 
of the cost of fi nd(), connected O, and uni on() is log N. 

Proof. Each operation does at most a constant number of array accesses for each 
node on the path from a node to a root in the forest. 


For dynamic connectivity, the practical implication of proposition h and its corollary 
is that weighted quick-union is the only one of the three algorithms that can feasibly 
be used for huge practical problems. The weighted quick-union algorithm uses at most 
cMlgN array accesses to process M connections among N sites for a small constant c. 
This result is in stark contrast to our finding that quick-find always (and quick-union 
sometimes) uses at least MN array accesses. Thus, with weighted quick-union, we can 
guarantee that we can solve huge practical dynamic connectivity problems in a reason¬ 
able amount of time. For the price of a few extra lines of code, we get a program that 
can be millions of times faster than the simpler algorithms for the huge dynamic con¬ 
nectivity problems that we might encounter in practical applications. 

A 100-site example is shown on the top of this page. It is evident from this diagram 
that relatively few nodes fall far from the root with weighted quick-union. Indeed it is 
frequently the case that a 1-node tree is merged with a larger tree, which puts the node 
just one link from the root. Empirical studies on huge problems tell us that weighted 
quick-union typically solves practical problems in constant time per operation. We 
could hardly expect to find a more efficient algorithm. 
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algorithm 


order of growth for N sites (worst case) 
constructor union find 


quick-find 

N 

N 

1 

quick-union 

N 

tree height 

tree height 

weighted quick-union 

N 

lg N 

lg N 

weighted quick-union with 
path compresson 

N 

very, very nearly, but not quite 1 (amc 
(see exercise 1.5.13) 

impossible 

N 

1 
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Performance characteristics of union-find algorithms 


Optimal algorithms. Can we find an algorithm that has guaranteed constant-time- 
per-operation performance? This question is an extremely difficult one that plagued 
researchers for many years. In pursuit of an answer, a number of variations of quick- 
union and weighted quick-union have been studied. For example, the following meth¬ 
od, known as path compression, is easy to implement. Ideally, we would like every node 
to link directly to the root of its tree, but we do not want to pay the price of changing a 
large number of links, as we did in the quick-find algorithm. We can approach the ideal 
simply by making all the nodes that we do examine directly link to the root. This step 
seems drastic at first blush, but it is easy to implement, and there is nothing sacrosanct 
about the structure of these trees: if we can modify them to make the algorithm more 
efficient, we should do so. To implement path compression, we just add another loop to 
f 1 i nd () that sets the i d [] entry corresponding to each node encountered along the way 
to link directly to the root. The net result is to flatten the trees almost completely, ap¬ 
proximating the ideal achieved by the quick-find algorithm. The method is simple and 
effective, but you are not likely to be able to discern any improvement over weighted 
quick-union in a practical situation (see exercise 1 . 5 . 24 ). Theoretical results about 
the situation are extremely complicated and quite remarkable. Weighted quick union 
with path compression is optimal but not quite constant-time per operation. That is, not 
only is weighted quick-find with path compression not constant-time per operation 
in the worst case (amortized), but also there exists no algorithm that can guarantee to 
perform each union-find operation in amortized constant time (under the very general 
“cell probe” model of computation). Weighted quick-union with path compression is 
very close to the best that we can do for this problem. 
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quick-find 


quick-union 

lOO-i 


0-1 


weighted quick-union 

20 —. 

0-1 - 


Amortized cost plots. As with any data type implementation, it is worthwhile to run 
experiments to test the validity of our performance hypotheses for typical clients, as dis¬ 
cussion in section 1.4. The figure at left shows 
details of the performance of the algorithms for 
our dynamic connectivity development client 
when solving our 625-site connectivity example 
(medi umUF.txt). Such diagrams are easy to pro¬ 
duce (see exercise 1 . 5 . 16 ): For the zth connec¬ 
tion processed, we maintain a variable cost that 
counts the number of array accesses (to i d [] or 
sz[]) and a variable total that is the sum of 
the total number of array accesses so far. Then 
we plot a gray dot at (i , cost) and a red dot 
at (i , total/i). The red dots are the average 
cost per operation, or amortized cost. These 
plots provide good insights into algorithm be¬ 
havior. For quick-find, every unionO opera¬ 
tion uses at least 625 accesses (plus 1 for each 
component merged, up to another 625) and 
every connected () operation uses 2 accesses. 
Initially, most of the connections lead to a call 
on unionO, so the cumulative average hovers 
around 625; later, most connections are calls to 
connected C) that cause the call to unionO to 
be skipped, so the cumulative average decreas¬ 
es, but still remains relatively high. (Inputs that 
lead to a large number of connected () calls that 
cause unionO to be skipped will exhibit signifi¬ 
cantly better performance—see exercise 1 . 5.23 
for an example). For quick-union, all operations 
initially require only a few array accesses; eventu¬ 
ally, the height of the trees becomes a significant 
factor and the amortized cost grows noticably. 
For weighted quick-union, the tree height stays 
small, none of the operations are expensive, and 
the amortized cost is low. These experiments 
validate our conclusion that weighted quick-union is certainly worth implementing 
and that there is not much further room for improvement for practical problems. 



find() operations 
become expensive ~~ 


20 




no expensive operations 

.. 


Cost of all operations (625 sites) 
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Perspective Each of the U F implementations that we considered is an improvement 
over the previous in some intuitive sense, but the process is artificially smooth because 
we have the benefit of hindsight in looking over the development of the algorithms as 
they were studied by researchers over the years. The implementations are simple and 
the problem is well specified, so we can evaluate the various algorithms directly by run¬ 
ning empirical studies. Furthermore, we can use these studies to validate mathematical 
results that quantify the performance of these algorithms. When possible, we follow the 
same basic steps for fundamental problems throughout the book that we have taken for 
union-find algorithms in this section, some of which are highlighted in this list: 

■ Decide on a complete and specific problem statement, including identifying 
fundamental abstract operations that are intrinsic to the problem and an API. 

■ Carefully develop a succinct implementation for a straightforward algorithm, 
using a well-thought-out development client and realistic input data. 

■ Know when an implementation could not possibly be used to solve problems on 
the scale contemplated and must be improved or abandoned. 

■ Develop improved implementations through a process of stepwise refinement, 
validating the efficacy of ideas for improvement through empirical analysis, 
mathematical analysis, or both. 

■ Find high-level abstract representations of data structures or algorithms in op¬ 
eration that enable effective high-level design of improved versions. 

■ Strive for worst-case performance guarantees when possible, but accept good 
performance on typical data when available. 

■ Know when to leave further improvements for detailed in-depth study to skilled 
researchers and move on to the next problem. 

The potential for spectacular performance improvements for practical problems such 
as those that we saw for union-find makes algorithm design a compelling field of study. 
What other design activities hold the potential to reap savings factors of millions or 
billions, or more? 

Developing an efficient algorithm is an intellectually satisfying activity that can have 
direct practical payoff. As the dynamic connectivity problem indicates, a simply stated 
problem can lead us to study numerous algorithms that are not only both useful and 
interesting, but also intricate and challenging to understand. We shall encounter many 
ingenious algorithms that have been developed over the years for a host of practical 
problems. As the scope of applicability of computational solutions to scientific and 
commercial problems widens, so also grows the importance of being able to use ef¬ 
ficient algorithms to solve known problems and of being able to develop efficient solu¬ 
tions to new problems. 
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Q. I’d like to add a del ete() method to the API that allows clients to delete connec¬ 
tions. Any advice on how to proceed? 

A. No one has devised an algorithm as simple and efficient as the ones in this section 
that can handle deletions. This theme recurs throughout this book. Several of the data 
structures that we consider have the property that deleting something is much more 
difficult than adding something. 

Q. What is the cell-probe model? 

A. A model of computation where we only count accesses to a random-access memory 
large enough to hold the input and consider all other operations to be free. 
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EXERCISES 

1.5.1 Show the contents of the id[] array and the number of times the ar¬ 
ray is accessed for each input pair when you use quick-find for the sequence 
9-0 3-4 5-8 7-2 2-1 5-7 0-3 4-2. 

1.5.2 Do exercise 1 . 5 . 1 ,but use quick-union (page 224). In addition, drawthe forest of 
trees represented by the i d [] array after each input pair is processed. 

1.5.3 Do exercise 1 . 5 . 1 , but use weighted quick-union (page 228). 

1.5.4 Show the contents of the sz [] and i d [] arrays and the number of array accesses 
for each input pair corresponding to the weighted quick-union examples in the text 
(both the reference input and the worst-case input). 

1.5.5 Estimate the minimum amount of time (in days) that would be required for 
quick-find to solve a dynamic connectivity problem with 10 9 sites and 10 6 input pairs, 
on a computer capable of executing 10 9 instructions per second. Assume that each itera¬ 
tion of the inner for loop requires 10 machine instructions. 

1.5.6 Repeat exercise 1 . 5.5 for weighted quick-union. 

1.5.7 Develop classes Qui ckUnionUF and Qui ckFi ndUF that implement quick-union 
and quick-find, respectively. 

1.5.8 Give a counterexample that shows why this intuitive implementation of uni on () 
for quick-find is not correct: 

public void uni on(int p, int q) 

{ 

if (connectedfp, q)) return; 

// Rename p’s component to q’s name. 

for (int i =0; i < id.length; i++) 
if Cid[i] == id[p]) id[i] = id[q]; 
count--; 

} 

1.5.9 Draw the tree corresponding to the i d [] array depicted at 

right. Can this be the result of running weighted quick-union? ^ 0123456789 

Explain why this is impossible or give a sequence of operations 1131561345 

that results in this array. 
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1.5.10 In the weighted quick-union algorithm, suppose that we set i d [fi nd (p) ] to q 
instead of to i d [f i nd (q) ]. Would the resulting algorithm be correct? 

Answer: Yes, but it would increase the tree height, so the performance guarantee would 
be invalid. 

1.5.11 Implement weighted quick-find, where you always change the id[] entries of 
the smaller component to the identifier of the larger component. How does this change 
affect performance? 
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CREATIVE PROBLEMS 

1 .5.12 Quick-union with path compression. Modify quick-union (page 224) to include 
path compression, by adding a loop to uni on() that links every site on the paths from 
p and q to the roots of their trees to the root of the new tree. Give a sequence of input 
pairs that causes this method to produce a path of length 4. Note: The amortized cost 
per operation for this algorithm is known to be logarithmic. 

1.5.13 Weighted quick-union with path compression. Modify weighted quick-union 
(algorithm 1 . 5 ) to implement path compression, as described in exercise 1 . 5 . 12 . 
Give a sequence of input pairs that causes this method to produce a tree of height 4. 
Note: The amortized cost per operation for this algorithm is known to be bounded by a 
function known as the inverse Ackermann function and is less than 5 for any conceivable 
practical value of N. 

1.5.14 Weighted quick-union by height. Develop a UF implementation that uses the 
same basic strategy as weighted quick-union but keeps track of tree height and always 
links the shorter tree to the taller one. Prove a logarithmic upper bound on the height 
of the trees for N sites with your algorithm. 

1.5.15 Binomial trees. Show that the number of nodes at each level in the worst-case 
trees for weighted quick-union are binomial coefficients. Compute the average depth of 
a node in a worst-case tree with N=2 n nodes. 

1.5.16 Amortized costs plots. Instrument your implementations from exercise 1 . 5.7 
to make amortized costs plots like those in the text. 

1.5.17 Random connections. Develop a UF client ErdosRenyi that takes an integer 
value N from the command line, generates random pairs of integers between 0 and N-l, 
calling connected () to determine if they are connected and then uni on() if not (as in 
our development client), looping until all sites are connected, and printing the number 
of connections generated. Package your program as a static method count O that takes 
N as argument and returns the number of connections and a mai n () that takes N from 
the command line, calls countf), and prints the returned value. 

1.5.18 Random grid generator. Write a program RandomGri d that takes an i nt value 
N from the command line, generates all the connections in an N-by-N grid, puts them 
in random order, randomly orients them (so that p q and q p are equally likely to oc¬ 
cur), and prints the result to standard output. To randomly order the connections, use 
a RandomBag (see exercise 1 . 3.34 on page 167). To encapsulate p and q in a single object, 


www.it-ebooks.info 





238 


CHAPTER 1 ■ Fundamentals 


CREATIVE PROBLEMS (continued) 


use the Connection nested class shown below. Package your program as two static 
methods: generateO, which takes N as argument and returns an array of connec¬ 
tions, and mai n(), which takes N from the command line, calls generateO, and iterates 
through the returned array to print the connections. 

1 . 5.19 Animation. Write a RandomGrid client (see exercise 1.5.18) that uses 
UnionFind as in our development client to check connectivity and uses StdDraw to 
draw the connections as they are processed. 

1 . 5.20 Dynamic growth. Using linked lists or a resizing array, develop a weighted 
quick-union implementation that removes the restriction on needing the number of 
objects ahead of time. Add a method newSiteO to the API, which returns an int 
identifier. 


private class Connection 
{ 

int p; 
int q; 

public Connection(int p, int q) 
{ this.p = p; this.q = q; } 


Record to encapsulate connections 
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| EXPERIMENTS 


1.5.21 Erdos-Renyi model. Use your client from exercise 1.5.17 to test the hypothesis 
that the number of pairs generated to get one component is ~ V 2 N In N. 

1.5.22 Doubling test for Erdos-Renyi model. Develop a performance-testing client that 
takes an i nt value T from the command line and performs T trials of the following ex¬ 
periment: Use your client from exercise 1.5.17 to generate random connections, using 
UnionFind to determine connectivity as in our development client, looping until all 
sites are connected. For each N, print the value of N, the average number of connections 
processed, and the ratio of the running time to the previous. Use your program to vali¬ 
date the hypotheses in the text that the running times for quick-find and quick-union 
are quadratic and weighted quick-union is near-linear. 

1.5.23 Compare quick-find with quick-union for Erdos-Renyi model. Develop a perfor¬ 
mance-testing client that takes an i nt value T from the command line and performs 
T trials of the following experiment: Use your client from exercise 1.5.17 to generate 
random connections. Save the connections, so that you can use both quick-union and 
quick-find to determine connectivity as in our development client, looping until all 
sites are connected. For each N, print the value of N and the ratio of the two running 
times. 


1.5.24 Fast algorithms for Erdos-Renyi model. Add weighted quick-union and weight¬ 
ed quick-union with path compression to your tests from exercise 1.5.23 . Can you 
discern a difference between these two algorithms? 

1.5.25 Doubling test for random grids. Develop a performance-testing client that takes 
an i nt value T from the command line and performs T trials of the following experie- 
ment: Use your client from exercise 1.5.18 to generate the connections in an N-by-N 
square grid, randomly oriented and in random order, then use Uni onFi nd to determine 
connectivity as in our development client, looping until all sites are connected. For each 
N, print the value of N, the average number of connections processed, and the ratio of 
the running time to the previous. Use your program to validate the hypotheses in the 
text that the running times for quick-find and quick-union are quadratic and weighted 
quick-union is near-linear. Note: As N doubles, the number of sites in the grid increases 
by a factor of 4 , so expect a doubling factor of 16 for quadratic and 4 for linear. 
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EXPERIMENTS (continued) 


1.5.26 Amortized plot for Erdds-Renyi. Develop a client that takes an i nt value N from 
the command line and does an amortized plot of the cost of all operations in the style 
of the plots in the text for the process of generating random pairs of integers between 0 
and N-l, calling connected () to determine if they are connected and then union() if 
not (as in our development client), looping until all sites are connected. 
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S orting is the process of rearranging a sequence of objects so as to put them in 
some logical order. For example, your credit card bill presents transactions in 
order by date—they were likely put into that order by a sorting algorithm. In the 
early days of computing, the common wisdom was that up to 30 percent of all com¬ 
puting cycles was spent sorting. If that fraction is lower today, one likely reason is that 
sorting algorithms are relatively efficient, not that sorting has diminished in relative 
importance. Indeed, the ubiquity of computer usage has put us awash in data, and the 
first step to organizing data is often to sort it. All computer systems have implementa¬ 
tions of sorting algorithms, for use by the system and by users. 

There are three practical reasons for you to study sorting algorithms, even though 
you might just use a system sort: 

■ Analyzing sorting algorithms is a thorough introduction to the approach that we 
use to compare algorithm performance throughout the book. 

■ Similar techniques are effective in addressing other problems. 

■ We often use sorting algorithms as a starting point to solve other problems. 

More important than these practical reasons is that the algorithms are elegant, classic, 
and effective. 

Sorting plays a major role in commercial data processing and in modern scientific 
computing. Applications abound in transaction processing, combinatorial optimiza¬ 
tion, astrophysics, molecular dynamics, linguistics, genomics, weather prediction, and 
many other fields. Indeed, a sorting algorithm (quicksort, in section 2.3) was named 
as one of the top ten algorithms for science and engineering of the 20th century. 

In this chapter, we consider several classical sorting methods and an efficient imple¬ 
mentation of a fundamental data type known as the priority queue. We discuss the 
theoretical basis for comparing sorting algorithms and conclude the chapter with a 
survey of applications of sorting and priority queues. 
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for our first excursion into the area of sorting algorithms, we shall study two ele¬ 
mentary sorting methods and a variation of one of them. Among the reasons for study¬ 
ing these relatively simple algorithms in detail are the following: First, they provide 
context in which we can learn terminology and basic mechanisms. Second, these simple 
algorithms are more effective in some applications than the sophisticated algorithms 
that we shall discuss later. Third, they are useful in improving the efficiency of more 
sophisticated algorithms, as we will see. 

Rules of the game Our primary concern is algorithms for rearranging arrays of 
items where each item contains a key. The objective of the sorting algorithm is to rear¬ 
range the items such that their keys are ordered according to some well-defined order¬ 
ing rule (usually numerical or alphabetical order). We want to rearrange the array so 
that each entry’s key is no smaller than the key in each entry with a lower index and 
no larger than the key in each entry with a larger index. Specific characteristics of the 
keys and the items can vary widely across applications. In Java, items are just objects, 
and the abstract notion of a key is captured in a built-in mechanism—the Comparabl e 
interface—that is described on page 247 . 

The class Exampl e on the facing page illustrates the conventions that we shall use: 
we put our sort code in a so rt () method within a single class along with private helper 
functions 1 ess() and exch() (and perhaps some others) and a sample client mai n(). 
Example also illustrates code that might be useful for initial debugging: its test client 
mai n () sorts strings from standard input using the private method show() to print the 
contents of the array. Later in this chapter, we will examine various test clients for com¬ 
paring algorithms and for studying their performance. To differentiate sorting meth¬ 
ods, we give our various sort classes different names. Clients can call different imple¬ 
mentations by name: Insert!on.sort(), Merge.sort (), Qui ck. sort (), and so forth. 

With but a few exceptions, our sort code refers to the data only through two opera¬ 
tions: the method 1 ess () that compares items and the method exch () that exchanges 
them. The exchO method is easy to implement, and the Comparabl e interface makes 
it easy to implement 1 ess (). Restricting data access to these two operations makes our 
code readable and portable, and makes it easier for us certify that algorithms are cor¬ 
rect, to study performance and to compare algorithms. Before proceeding to consider 
sort implementations, we discuss a number of important issues that need to be care¬ 
fully considered for every sort. 
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Template for sort classes 


public class Example 

{ 

public static void sort(Comparable[] a) 

{ /* See Algorithms 2.1, 2.2, 2.3, 2.4, 2.5, or 2.7. */ } 

private static boolean less(Comparable v, Comparable w) 

{ return v.compareTo(w) <0; } 

private static void exch(Comparable[] a, int i, int j) 

{ Comparable t = a[i]; a[i] = a[j]; a[j] = t; } 

private static void show(Comparable[] a) 

{ // Print the array, on a single line. 

for (int i = 0; i < a.length; i++) 

StdOut.print(a[i] + " "); 

StdOut.printlnO ; 

} 

public static boolean isSorted(Comparable[] a) 

{ // Test whether the array entries are in order. 

for (int i = 1; i < a.length; i++) 

if (less(a[i], a[i-l])) return false; 
return true; 

} 


public static void main(String[] args) 

{ // Read strings from standard input, sort them, and print. 


String[] a = In.readStrings() 
sort(a); 

assert isSorted(a); 
show(a); 

} 

} 


This class illustrates our conventions for imple¬ 
menting array sorts. For each sorting algorithm 
that we consider, we present a sort() method for 
a class like this with Exampl e changed to a name 
that corresponds to the algorithm. The test client 
sorts strings taken from standard input, but, with 
this code, our sort methods are effective for any 
type of data that implements Comparabl e. 


% more tiny.txt 
SORTEXAMPLE 

% java Example < tiny.txt 
AEELMOPRSTX 


% more words3.txt 

bed bug dad yes zoo ... all bad yet 

% java Example < words.txt 

all bad bed bug dad ... yes yet zoo 
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Certification. Does the sort implementation always put the array in order, no mat¬ 
ter what the initial order? As a conservative practice, we include the statement 
assert i sSorted (a) ; in our test client to certify that array entries are in order after 
the sort. It is reasonable to include this statement in every sort implementation, even 
though we normally test our code and develop mathematical arguments that our al¬ 
gorithms are correct. Note that this test is sufficient only if we use exch() exclusively 
to change array entries. When we use code that stores values into the array directly, we 
do not have full assurance (for example, code that destroys the original input array by 
setting all values to be the same would pass this test). 

Running time. We also test algorithm performance. We start by 
proving facts about the number of basic operations (compares 
and exchanges, or perhaps the number of times the array is ac¬ 
cessed, for read or write) that the various sorting algorithms per¬ 
form for various natural input models. Then we use these facts 
to develop hypotheses about the comparative performance of the 
algorithms and present tools that you can use to experimentally 
check the validity of such hypotheses. We use a consistent coding 
style to facilitate the development of valid hypotheses about per¬ 
formance that will hold true for typical implementations. 

Extra memory. The amount of extra memory used by a sorting algorithm is often as 
important a factor as running time. The sorting algorithms divide into two basic types: 
those that sort in place and use no extra memory except perhaps for a small function- 
call stack or a constant number of instance variables, and those that need enough extra 
memory to hold another copy of the array to be sorted. 

Types of data. Our sort code is effective for any item type that implements the 
Comparable interface. Adhering to Java’s convention in this way is convenient be¬ 
cause many of the types of data that you might want to sort implement Comparabl e. 
For example, Java’s numeric wrapper types such as Integer and Double implement 
Comparable, as do String and various advanced types such as File or URL. Thus, 
you can just call one of our sort methods with an array of any of these types as argu¬ 
ment. For example, the code at right uses quicksort (see section 2.3) to sort N random 
Double values. When we create types of our 
own, we can enable client code to sort that type 
of data by implementing the Comparable in¬ 
terface. To do so, we just need to implement a 
compareToO method that defines an ordering 

on objects of that type known as the natural Sorting an array of random values 


Double a[] = new Double[N]; 
for (int i = 0; i < N; i++) 
a [i ] = StdRandom. uniformO ; 
Quick, sort(a) ; 


Sorting cost model. 

When studying sorting 
algorithms, we count 
compares and exchanges. 
For algorithms that do 
not use exchanges, we 
count array accesses. 
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order for that type, as shown here for our Date data type (see page 91 ). Java’s convention 
is that the call v. compareTo(w) returns an integer that is negative, zero, or positive 
(usually - 1 , 0, or +1) when v<w, v = w, 
or v>w, respectively. For economy, we 
use standard notation like v>w as short¬ 
hand for code like v.compareTo(w)>0 
for the remainder of this paragraph. By 
convention, v.compareTo(w) throws 
an exception if v and w are incompatible 
types or either is null. Furthermore, 
compareToO must implement a total 
order, it must be 

■ Reflexive (for all v, v = v) 

■ Antisymmetric (for all v and w, if 
v<w then w> v and if v = w then 
w=v) 

■ Transitive (for all v, w, and x, if 
v<= w and w<=x then v<=x) 

These rules are intuitive and standard 
in mathematics—you will have little 
difficulty adhering to them. In short, 
compareToO implements our key ab¬ 
straction—it defines the ordering of 
the items (objects) to be sorted, which 
can be any type of data that implements 
Comparabl e. Note that compareToO need not use all of the instance variables. Indeed, 
the key might be a small part of each item. 


public class Date implements Comparable<Date> 
{ 

private final int day; 
private final int month; 
private final int year; 

public Date (int d, int m, int y) 

{ day = d; month = m; year = y; } 

public int day() { return day; } 

public int month() { return month; } 

public int year() { return year; } 

public int compareTo(Date that) 

{ 

if (this.year > that.year ) return +1; 

if (this.year < that.year ) return -1; 

if (this.month > that.month) return +1; 
if (this.month < that.month) return -1; 
if (this.day > that.day ) return +1; 

if (this.day < that.day ) return -1; 

return 0; 

} 

public String toStringO 

{ return month + "/" + day + "/" + year; } 


Defining a comparable type 


for the remainder of this chapter, we shall address numerous algorithms for sort¬ 
ing arrays of objects having a natural order. To compare and contrast the algorithms, 
we shall examine a number of their properties, including the number of compares and 
exchanges that they use for various types of inputs and the amount of extra memory 
that they use. These properties lead to the development of hypotheses about perfor¬ 
mance properties, many of which have been validated on countless computers over the 
past several decades. Specific implementations always need to be checked, so we also 
consider tools for doing so. After considering the classic selection sort, insertion sort, 
shellsort, mergesort, quicksort, and heapsort algorithms, we will consider practical is¬ 
sues and applications, in section 2.5. 
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Selection sort One of the simplest sorting algorithms works as follows: First, find 
the smallest item in the array and exchange it with the first entry (itself if the first entry 
is already the smallest). Then, find the next smallest item and exchange it with the sec¬ 
ond entry. Continue in this way until the entire array is sorted. This method is called 
selection sort because it works by repeatedly selecting the smallest remaining item. 

As you can see from the implementation in algorithm 2.1, the inner loop of selec¬ 
tion sort is just a compare to test a current item against the smallest item found so far 
(plus the code necessary to increment the current index and to check that it does not 
exceed the array bounds); it could hardly be simpler. The work of moving the items 
around falls outside the inner loop: each exchange puts an item into its final position, 
so the number of exchanges is N. Thus, the running time is dominated by the number 
of compares. 

Proposition A. Selection sort uses ~N 2 /2 compares and N exchanges to sort an 
array of length N. 

Proof: You can prove this fact by examining the trace, which is an iV-by-iV table 
in which unshaded letters correspond to compares. About one-half of the entries 
in the table are unshaded—those on and above the diagonal. The entries on the 
diagonal each correspond to an exchange. More precisely, examination of the code 
reveals that, for each i from 0 to N — 1 , there is one exchange and N — 1 — i com¬ 
pares, so the totals are N exchanges and (N — 1 ) + (N — 2 )+ ... + 2 + 1+0 = N(N 
— 1 ) / 2 ~ N 2 / 2 compares. 


in summary, selection sort is a simple sorting method that is easy to understand and to 
implement and is characterized by the following two signature properties: 

Running time is insensitive to input. The process of finding the smallest item on one 
pass through the array does not give much information about where the smallest item 
might be on the next pass. This property can be disadvantageous in some situations. 
For example, the person using the sort client might be surprised to realize that it takes 
about as long to run selection sort for an array that is already in order or for an array 
with all keys equal as it does for a randomly-ordered array! As we shall see, other algo¬ 
rithms are better able to take advantage of initial order in the input. 

Data movement is minimal. Each of the N exchanges changes the value of two array 
entries, so selection sort uses N exchanges—the number of array accesses is a linear 
function of the array size. None of the other sorting algorithms that we consider have 
this property (most involve linearithmic or quadratic growth). 
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ALGORITHM 2.1 Selection sort 


public class Selection 

{ 

public static void sortCComparable[] a) 

{ // Sort a[] into increasing order. 

int N = a.length; // array length 

for (int i =0; i < N; i++) 

{ // Exchange a[i] with smallest entry in a[i+l...N). 

int min = i; // index of minimal entr. 

for (int j = i+1; j < N; j++) 

if (less(a[j], a[min])) min = j; 
exch(a, i, min); 

} 

} 

// See page 245 for less(), exch(), isSortedQ, and main(). 


For each i, this implementation puts the i th smallest item in a [i ]. The entries to the left of position 
i are the i smallest items in the array and are not examined again. 
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Trace of selection sort (array contents just after each exchange) 
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Insertion sort The algorithm that people often use to sort bridge hands is to con¬ 
sider the cards one at a time, inserting each into its proper place among those already 
considered (keeping them sorted). In a computer implementation, we need to make 
space to insert the current item by moving larger items one position to the right, before 
inserting the current item into the vacated position, algorithm 2.2 is an implementa¬ 
tion of this method, which is called insertion sort. 

As in selection sort, the items to the left of the current index are in sorted order dur¬ 
ing the sort, but they are not in their final position, as they may have to be moved to 
make room for smaller items encountered later. The array is, however, fully sorted when 
the index reaches the right end. 

Unlike that of selection sort, the running time of insertion sort depends on the ini¬ 
tial order of the items in the input. For example, if the array is large and its entries are 
already in order (or nearly in order), then insertion sort is much, much faster than if 
the entries are randomly ordered or in reverse order. 

Proposition B. Insertion sort uses ~N 2 /4 compares and ~N 2 /4 exchanges to sort 
a randomly ordered array of length N with distinct keys, on the average. The worst 
case is ~N 2 /2 compares and ~N 2 /2 exchanges and the best case isN — 1 compares 
and 0 exchanges. 

Proof: Just as for proposition a, the number of compares and exchanges is easy to 
visualize in the N-by-N diagram that we use to illustrate the sort. We count entries 
below the diagonal—all of them, in the worst case, and none of them, in the best 
case. For randomly ordered arrays, we expect each item to go about halfway back, 
on the average, so we count one-half of the entries below the diagonal. 

The number of compares is the number of exchanges plus an additional term 
equal to N minus the number of times the item inserted is the smallest so far. In the 
worst case (array in reverse order), this term is negligible in relation to the total; in 
the best case (array in order) it is equal to N — 1 . 


Insertion sort works well for certain types of nonrandom arrays that often arise in 
practice, even if they are huge. For example, as just mentioned, consider what happens 
when you use insertion sort on an array that is already sorted. Each item is immediately 
determined to be in its proper place in the array, and the total running time is linear. 
(The running time of selection sort is quadratic for such an array.) The same is true 
for arrays whose keys are all equal (hence the condition in proposition b that the keys 
must be distinct). 
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ALGORITHM 2.2 Insertion sort 


public class Insertion 

{ 

public static void sortCComparable[] a) 

{ // Sort a[] into increasing order. 

int N = a.length; 

for (int i = 1; i < N; i++) 

{ // Insert a[i] among a[i-l], a[i-2], a[i-3]. 

for (int j = i; j > 0 && less(a[j], a[j-l]); j--) 
exch(a, j, j-1); 

} 

} 

// See page 245 for less(), exch(), isSorted(), and main(). 

} 


For each i from 0 to N-l, exchange a[i ] with the entries that are smaller in a[ 0 ] through a[i -1]. As 
the index i travels from left to right, the entries to its left are in sorted order in the array, so the array 
is fully sorted when i reaches the right end. 
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Trace of insertion sort (array contents just after each insertion) 
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More generally, we consider the concept of a partially sorted array, as follows: An in¬ 
version is a pair of entries that are out of order in the array. For instance, EXAMPLE 
has 11 inversions: E-A, X-A, X-M, X-P, X-L, X-E, M-L, M-E, P-L, P-E,and L-E.Ifthe 
number of inversions in an array is less than a constant multiple of the array size, we 
say that the array is partially sorted. Typical examples of partially sorted arrays are the 
following: 

■ An array where each entry is not far from its final position 

■ A small array appended to a large sorted array 

■ An array with only a few entries that are not in place 

Insertion sort is an efficient method for such arrays; selection sort is not. Indeed, when 
the number of inversions is low, insertion sort is likely to be faster than any sorting 
method that we consider in this chapter. 

Proposition C. The number of exchanges used by insertion sort is equal to the 
number of inversions in the array, and the number of compares is at least equal to 
the number of inversions and at most equal to the number of inversions plus the 
array size minus 1. 

Proof: Every exchange involves two inverted adjacent entries and thus reduces the 
number of inversions by one, and the array is sorted when the number of inver¬ 
sions reaches zero. Every exchange corresponds to a compare, and an additional 
compare might happen for each value of i from 1 to N-l (when a[i] does not 
reach the left end of the array). 


It is not difficult to speed up insertion sort substantially, by shortening its inner loop to 
move the larger entries to the right one position rather than doing full exchanges (thus 
cutting the number of array accesses in half). We leave this improvement for an exercise 
(see exercise 2.1.25). 

in summary, insertion sort is an excellent method for partially sorted arrays and is also 
a fine method for tiny arrays. These facts are important not just because such arrays 
frequently arise in practice, but also because both types of arrays arise in intermediate 
stages of advanced sorting algorithms, so we will be considering insertion sort again in 
relation to such algorithms. 
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Visualizing sorting algorithms Throughout this chapter, we will be using 
a simple visual representation to help describe the properties of sorting algorithms. 
Rather than tracing the progress of a sort with key values such as letters, numbers, 
or words, we use vertical bars, to be sorted by their 
heights. The advantage of such a representation is 
that it can give insights into the behavior of a sort¬ 
ing method. 

For example, you can see at a glance on the visual 
traces at right that insertion sort does not touch 
entries to the right of the scan pointer and selec¬ 
tion sort does not touch entries to the left of the 
scan pointer. Moreover, it is clear from the visual 
traces that, since insertion sort also does not touch 
entries smaller than the inserted item, it uses about 
half the number of compares as selection sort, on 
the average. 

With our StdDraw library, developing a visual 
trace is not much more difficult than doing a stan¬ 
dard trace. We sort Double values, instrument the 
algorithm to call show() as appropriate (just as we 
do for a standard trace), and develop a version of 
show() that uses StdDraw to draw the bars instead 
of printing the results. The most complicated task 
is setting the scale for the y-axis so that the lines of 
the trace appear in the expected order. You are en¬ 
couraged to work exercise 2.1.18 in order to gain a 
better appreciation of the value of visual traces and 
the ease of creating them. 

An even simpler task is to animate the trace so 
that you can see the array dynamically evolve to 
the sorted result. Developing an animated trace in¬ 
volves essentially the same process described in the previous paragraph, but without 
having to worry about the y-axis (just clear the window and redraw the bars each time). 
Though we cannot make the case on the printed page, such animated representations 
are also effective in gaining insight into how an algorithm works. You are also encour¬ 
aged to work exercise 2.1.17 to see for yourself. 
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Comparing two sorting algorithms Now that we have two implementations, 
we are naturally interested in knowing which one is faster: selection sort (algorithm 
2.1) or insertion sort (algorithm 2.2). Questions like this arise again and again and 
again in the study of algorithms and are a major focus throughotit this book. We have 
discussed some fundamental ideas in chapter 1, but we use this first case in point to 
illustrate our basic approach to answering such questions. Generally, following the ap¬ 
proach introduced in section 1.4, we compare algorithms by 

■ Implementing and debugging them 

■ Analyzing their basic properties 

■ Formulating a hypothesis abotit comparative performance 

■ Running experiments to validate the hypothesis 

These steps are nothing more than the time-honored scientific method, applied to the 
study of algorithms. 

In the present context, algorithm 2.1 and algorithm 2.2 are evidence of the 
first step; propositions a, b, and c constitute the second step; property d on page 255 
constitutes the third step; and the class So rtCompa re on page 256 enables the fourth step. 
These activities are all interrelated. 

Our brief descriptions mask a substantial amount of effort that is required to prop¬ 
erly implement, analyze, and test algorithms. Every programmer knows that such code 
is the product of a long round of debugging and refinement, every mathematician 
knows that proper analysis can be very difficult, and every scientist knows that formu¬ 
lating hypotheses and designing and executing experiments to validate them require 
great care. Full development of such results is reserved for experts studying our most 
important algorithms, but every programmer using an algorithm should be aware of 
the scientific context underlying its performance properties. 

Having developed implementations, our next choice is to settle on an appropriate 
model for the input. For sorting, a natLiral model, which we have used for proposi¬ 
tions a, b, and c, is to assume that the arrays are randomly ordered and that the key 
values are distinct. In applications where significant numbers of equal key values are 
present we will need a more complicated model. 

How do we formulate a hypothesis about the running times of insertion sort and 
selection sort for randomly ordered arrays? Examining algorithms 2.1 and 2.2 and 
propositions A and b, it follows immediately that the running time of both algorithms 
should be quadratic for randomly ordered arrays. That is, the running time of insertion 
sort for such an input is proportional to some small constant times N 2 and the running 
time of selection sort is proportional to some other small constant times N 2 . The values 
of the two constants depend on the cost of compares and exchanges on the particular 
computer being used. For many types of data and for typical computers, it is reasonable 
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to assume that these costs are similar (though we will see a few significant exceptions) 
The following hypothesis follows directly: 

Property D. The running times of insertion sort and selection sort are quadratic 
and within a small constant factor of one another for randomly ordered arrays of 
distinct values. 

Evidence: This statement has been validated on many different computers over 
the past half-century. Insertion sort was about twice as fast as selection sort when 
the first edition of this book was written in 1980 and it still is today, even though it 
took several hours to sort 100,000 items with these algorithms then and just several 
seconds today. Is insertion sort a bit faster than selection sort on your computer? 
To find out, you can use the class SortCompare on the next page, which uses the 
sort() methods in the classes named as command-line arguments to perform the 
given number of experiments (sorting arrays of the given size) and prints the ratio 
of the observed running times of the algorithms. 


To validate this hypothesis, we use SortCompare (see page 256 ) to perform the experi¬ 
ments. As usual, we use Stopwatch to compute the running time. The implementation 
of ti me () shown here does the job for the basic sorts in this chapter. The “randomly or¬ 
dered” input model is embedded in the timeRandomlnputO method in SortCompare, 
which generates random Double values, sorts them, and returns the total measured 
time of the sort for the given 
number of trials. Using ran¬ 
dom Double values between 
0.0 and 1.0 is much simpler 
than the alternative of us¬ 
ing a library function such 
as StdRandom.shuffleO 
and is effective because equal 
key values are very unlikely 
(see exercise 2.5.31). As 
discussed in chapter 1, the 
number of trials is taken as an 
argument both to take advan¬ 
tage of the law of large numbers (the more trials, the total running time divided by the 
number of trials is a more accurate estimate of the true average running time) and to 
help damp out system effects. You are encouraged to experiment with SortCompare 


public static double time(String alg, Comparable[] a) 
{ 

Stopwatch timer = new Stopwatch(); 
if (alg.equals("Insertion")) Insertion . sort(a) ; 
if (alg.equals("Selection")) Selection . sort(a) ; 
if Calg.equals("Shell")) Shel 1 . sort(a) ; 
if (alg.equals("Merge")) Merge. sort(a) ; 

if (alg.equals("Quick")) Qui ck. sort(a) ; 
if (alg.equals("Heap")) Heap. sort(a) ; 

return timer.elapsedTimeO ; 


Timing one of the sort algorithms in this chapter on a given input 
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Comparing two sorting algorithms 


public class SortCompare 

{ 

public static double time(String alg, Doublet] a) 

{ /* See text. */ } 

public static double timeRandomlnputCString alg, int N, int T) 

{ // Use alg to sort T random arrays of length N. 

double total = 0.0; 

Doublet] a = new DoubletN]; 
for (int t = 0; t < T; t++) 

{ // Perform one experiment (generate and sort an array). 

for (int i =0; i < N; i++) 
a(i] = StdRandom.uniformO ; 
total += time(alg, a); 

} 

return total; 

} 

public static void main(Stringt] args) 

{ 

String algl = argstO]; 

String alg2 = args[l]; 

int N = Integer.parselnt(argst2]); 

int T = Integer.parselnt(argstB]); 

double tl = timeRandomInput(algl, N, T); // total for algl 
double t2 = timeRandomInput(alg2, N, T); // total for alg2 
StdOut.printf("For %d random Doubles\n %s is", N, algl); 
StdOut.printf(" %.lf times faster than %s\n", t2/tl, alg2); 

} 

} 


This client runs the two sorts named in the first two command-line arguments on arrays of N (the 
third command-line argument) random Double values between 0.0 and 1.0, repeating the experi¬ 
ment T (the fourth command-line argument) times, then prints the ratio of the total running times. 

% java SortCompare Insertion Selection 1000 100 
For 1000 random Doubles 

Insertion is 1.7 times faster than Selection 
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on your computer to learn the extent to which its conclusion about insertion sort and 
selection sort is robust. 

property d is intentionally a bit vague—the value of the small constant factor is left 
unstated and the assumption that the costs of compares and exchanges are similar is left 
unstated—so that it can apply in a broad variety of situations. When possible, we try to 
capture essential aspects of the performance of each of the algorithms that we study in 
statements like this. As discussed in chapter i, each Property that we consider needs to 
be tested scientifically in a given situation, perhaps supplemented with a more refined 
hypothesis based upon a related Proposition (mathematical truth). 

For practical applications, there is one further step, which is crucial: run experiments 
to validate the hypothesis on the data at hand. We defer consideration of this step to 
section 2.5 and the exercises. In this case, if your sort keys are not distinct and/or 
not randomly ordered, property d might not hold. You can randomly order an array 
with StdRandom. shuffleO, but applications with significant numbers of equal keys 
involve more careful analysis. 

Our discussions of the analyses of algorithms are intended to be starting points, not 
final conclusions. If some other question about performance of the algorithms comes 
to mind, you can study it with a tool like SortCompare. Many opportunities to do so 
are presented in the exercises. 

we do not dwell further on the comparative performance of insertion sort and selec¬ 
tion sort because we are much more interested in algorithms that can run a hundred or 
a thousand or a million times faster than either. Still, understanding these elementary 
algorithms is worthwhile for several reasons: 

■ They help us work out the ground rules. 

■ They provide performance benchmarks. 

■ They often are the method of choice in some specialized situations. 

■ They can serve as the basis for developing better algorithms. 

For these reasons, we will use the same basic approach and consider elementary algo¬ 
rithms for every problem that we study throughout this book, not just sorting. Pro¬ 
grams like SortCompare play a critical role in this incremental approach to algorithm 
development. At every step along the way, we can use such a program to help evaluate 
whether a new algorithm or an improved version of a known algorithm provides the 
performance gains that we expect. 
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Shellsort To exhibit the value of knowing properties of elementary sorts, we next 
consider a fast algorithm based on insertion sort. Insertion sort is slow for large un¬ 
ordered arrays because the only exchanges it does involve adjacent entries, so items 
can move through the array only one place at a time. For example, if the item with the 
smallest key happens to be at the end of the array, N— 1 exchanges are needed to get that 
one item where it belongs. Shellsort is a simple extension of insertion sort that gains 
speed by allowing exchanges of array entries that are far apart, to produce partially 
sorted arrays that can be efficiently sorted, eventually by insertion sort. 

The idea is to rearrange the array to give it the property that taking every //th entry 
(starting anywhere) yields a sorted subsequence. Such an array is said to be //-sorted. Put 

another way, an h- sorted array is h inde- 
h ~ 4 pendent sorted subsequences, interleaved 

L _ M _ p _ T together. By //-sorting tor some large val- 

E _ H _ s _ s ues of h, we can move items in the array 

E - L - 0 - X long distances and thus make it easier to 

A-E-L-R //-sort for smaller values of h. Using such 

a procedure for any sequence of values of 

An fi-sorted sequence is h interleaved sorted subsequences . . . . ... . . 

// that ends m 1 will produce a sorted ar¬ 
ray: that is shellsort. The implementation 
in algorithm 2.3 on the facing page uses the sequence of decreasing values V 2 ( 3 k — 1), 
starting at the largest increment less than NI 3 and decreasing to 1 . We refer to such a 
sequence as an increment sequence, algorithm 2.3 computes its increment sequence; 
another alternative is to store an increment sequence in an array. 

One way to implement shellsort would be, for each h, to use insertion sort indepen¬ 
dently on each of the h subsequences. Because the subsequences are independent, we 
can use an even simpler approach: when h- sorting the array, we insert each item among 
the previous items in its //-subsequence by exchanging it with those that have larger 
keys (moving them each one position to the right in the subsequence). We accomplish 
this task by using the insertion-sort code, but modified to decrement by h instead of 1 
when moving through the array. This observation reduces the shellsort implementa¬ 
tion to an insertion-sort-like pass through the array for each increment. 

Shellsort gains efficiency by making a tradeoff between size and partial order in the 
subsequences. At the beginning, the subsequences are short; later in the sort, the subse¬ 
quences are partially sorted. In both cases, insertion sort is the method of choice. The 
extent to which the subsequences are partially sorted is a variable factor that depends 
strongly on the increment sequence. Understanding shellsort’s performance is a chal¬ 
lenge. Indeed, algorithm 2.3 is the only sorting method we consider whose perfor¬ 
mance on randomly ordered arrays has not been precisely characterized. 
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ALGORITHM 2.3 Shellsort 


public class Shell 

{ 

public static void sortCComparable[] a) 

{ // Sort a[] into increasing order, 

int N = a.length; 
int h = 1; 

while (h < N/3) h = 3*h + 1; // 1, 4, 13, 40, 121, 364, 1093, ... 
while (h >= 1) 

{ // h-sort the array. 

for (int i = h; i < N; i++) 

{ // Insert a[i] among a[i—h], a[i-2*h], a[i-3*h]... . 

for (int j = i; j >= h && less(a[j], a[j-h]); j -= h) 
exch(a, j, j-h); 

} 

h = h/3; 

} 

} 

// See page 245 for less(), exch(), isSorted(), and main(). 

} 


If we modify insertion sort (algorithm 2.2) to h-sort the array and add an outer loop to decrease 
h through a sequence of increments starting at an increment as large as a constant fraction of the ar¬ 
ray length and ending at 1, we are led to this compact shellsort implementation. 


% java SortCompare Shell Insertion 100000 100 
For 100000 random Doubles 

Shell is 600 times faster than Insertion 
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Shellsort trace (array contents after each pass) 
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input 

13-sort 

4-sort 


1-sort 


result 


How do we decide what increment sequence to use? In general, this question is a dif¬ 
ficult one to answer. The performance of the algorithm depends not just on the num¬ 
ber of increments, but also on arithmetical interactions among the increments such as 

the size of their common divi- 
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sors and other properties. Many 
different increment sequences 
have been studied in the lit¬ 
erature, but no provably best 
sequence has been found. The 
increment sequence that is used 
in algorithm 2.3 is easy to 
compute and use, and performs 
nearly as well as more sophisti¬ 
cated increment sequences that 
have been discovered that have 
provably better worst-case per¬ 
formance. Increment sequences 
that are substantially better still 
may be waiting to be discovered. 

Shellsort is useful even for 
large arrays, particularly by 
contrast with selection sort and 
insertion sort. It also performs 
well on arrays that are in arbi¬ 
trary order (not necessarily ran¬ 
dom). Indeed, constructing an 
array for which shellsort runs 
slowly for a particular incre¬ 
ment sequence is usually a chal¬ 
lenging exercise. 

As you can learn with 
SortCompare, shellsort is much 
faster than insertion sort and 
selection sort, and its speed ad¬ 
vantage increases with the array 
size. Before reading further, try using SortCompare to compare shellsort with insertion 
sort and selection sort for array sizes that are increasing powers of 2 on your computer 
(see exercise 2.1.27). You will see that shellsort makes it possible to address sorting 


X R 
X R 
X R 
X R 
X R 


X R 


Detailed trace of shellsort (insertions) 
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input 






result 



Visual trace of shellsort 


problems that could not be addressed with the more elementary algorithms. This ex¬ 
ample is our first practical illustration of an important principle that pervades this 
book: achieving speedups that enable the solution of problems that could not otherwise be 
solved is one of the prime reasons to study algorithm performance and design. 

The study of the performance characteristics of shellsort requires mathematical ar¬ 
guments that are beyond the scope of this book. If you want to be convinced, start 
by thinking about how you would prove the following fact: when an h-sorted array is 
k-sorted, it remains h-sorted. As for the performance of algorithm 2.3, the most im¬ 
portant result in the present context is the knowledge that the running time of shellsort 
is not necessarily quadratic —for example, it is known that the worst-case number of 
compares for algorithm 2.3 is proportional to ;Y 3/2 . That such a simple modification 
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can break the quadratic-running-time barrier is quite interesting, as doing so is a prime 
goal for many algorithm design problems. 

No mathematical results are available about the average-case number of compares 
for shellsort for randomly ordered input. Increment sequences have been devised that 
drive the asymptotic growth of the worst-case number of compares down to N il3 , N 514 , 
N 615 , . . . , but many of these results are primarily of academic interest because these 
functions are hard to distinguish from one another (and from a constant factor of N) 
for practical values of N. 

In practice, you can safely take advantage of the past scientific study of shellsort just 
by using the increment sequence in algorithm 2.3 (or one of the increment sequences 
in the exercises at the end of this section, which may improve performance by 20 to 40 
percent). Moreover, you can easily validate the following hypothesis: 

Property E. The number of compares used by shellsort with the increments 1 , 4 , 
13 , 40 , 121 , 364 ,... is bounded by a small multiple of N times the number of incre¬ 
ments used. 

Evidence: Instrumenting algorithm 2.3 to count compares and divide by the 
number of increments used is a straightforward exercise (see exercise 2.1.12). Ex¬ 
tensive experiments suggest that the average number of compares per increment 
might be N 115 , but it is quite difficult to discern the growth in that function unless 
N is huge. This property also seems to be rather insensitive to the input model. 


experienced programmers sometimes choose shellsort because it has acceptable 
running time even for moderately large arrays; it requires a small amount of code; and 
it uses no extra space. In the next few sections, we shall see methods that are more ef¬ 
ficient, but they are perhaps only twice as fast (if that much) except for very large N, and 
they are more complicated. If you need a solution to a sorting problem, and are work¬ 
ing in a situation where a system sort may not be available (for example, code destined 
for hardware or an embedded system), you can safely use shellsort, then determine 
sometime later whether it will be worthwhile to replace it with a more sophisticated 
method. 
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Q&A 


Q. Sorting seems like a toy problem. Aren’t many of the other things that we do with 
computers much more interesting? 

A. Perhaps, but many of those interesting things are made possible by fast sorting al¬ 
gorithms. You will find many examples in section 2.5 and throughout the rest of the 
book. Sorting is worth studying now because the problem is easy to understand, and 
you can appreciate the ingenuity behind the faster algorithms. 

Q. Why so many sorting algorithms? 

A. One reason is that the performance of many algorithms depends on the input val¬ 
ues, so different algorithms might be appropriate for different applications having dif¬ 
ferent kinds of input. For example, insertion sort is the method of choice for partially 
sorted or tiny arrays. Other constraints, such as space and treatment of equal keys, also 
come into play. We will revisit this question in section 2 . 5 . 

Q. Why bother using the tiny helper methods 1 ess () and exch ()? 

A. They are basic abstract operations needed by any sort algorithm, and the code is 
easier to understand in terms of these abstractions. Moreover, they make the code di¬ 
rectly portable to other settings. For example, much of the code in algorithms 2.1 
and 2.2 is legal code in several other programming languages. Even in Java, we can use 
this code as the basis for sorting primitive types (which are not Comparable): simply 
implement 1 ess 0 with the code v < w. 

Q. When I run SortCompare, I get different values each time that I run it (and those 
are different from the values in the book). Why? 

A. For starters, you have a different computer from the one we used, not to mention 
a different operating system, Java runtime, and so forth. All of these differences might 
lead to slight differences in the machine code for the algorithms. Differences each time 
that you run it on your computer might be due to other applications that you are run¬ 
ning or various other conditions. Running a very large number of trials should dampen 
the effect. The lesson is that small differences in algorithm performance are difficult to 
notice nowadays. That is a primary reason that we focus on large ones! 
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EXERCISES 

2 . 1.1 Show, in the style of the example trace with algorithm 2.1, how selection sort 
sorts the array EASYQUESTION. 

2 . 1.2 What is the maximum number of exchanges involving any particular element 
during selection sort? What is the average number of exchanges involving an element? 

2 . 1.3 Give an example of an array of N items that maximizes the number of times the 
test a [ j ] < a [min] fails (and, therefore, min gets updated) during the operation of 
selection sort (algorithm 2.1). 

2 . 1.4 Show, in the style of the example trace with algorithm 2.2, how insertion sort 
sorts the array EASYQUESTION. 

2 . 1.5 For each of the two conditions in the inner for loop in insertion sort (algo¬ 
rithm 2.2), describe an array of N items where that condition is always false when the 
loop terminates. 

2 . 1.6 Which method runs faster for an array with all keys identical, selection sort or 
insertion sort? 

2 . 1.7 Which method runs faster for an array in reverse order, selection sort or inser¬ 
tion sort? 

2 . 1.8 Suppose that we use insertion sort on a randomly ordered array where elements 
have only one of three values. Is the running time linear, quadratic, or something in 
between? 

2 . 1.9 Show, in the style of the example trace with algorithm 2.3, how shellsort sorts 
the array EASYSHELLSORTQUESTION. 

2 . 1.10 Why not use selection sort for /(-sorting in shellsort? 

2 . 1.11 Implement a version of shellsort that keeps the increment sequence in an array, 
rather than computing it. 

2 . 1.12 Instrument shellsort to print the number of compares divided by the array size 
for each increment. Write a test client that tests the hypothesis that this number is a 
small constant, by sorting arrays of random Double values, using array sizes that are 
increasing powers of 10, starting at 100. 
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CREATIVE PROBLEMS 


2.1.13 Deck sort. Explain how you would put a deck of cards in order by suit (in the 
order spades, hearts, clubs, diamonds) and by rank within each suit, with the restriction 
that the cards must be laid out face down in a row, and the only allowed operations are 
to check the values of two cards and to exchange two cards (keeping them face down). 

2.1.14 Dequeue sort. Explain how you would sort a deck of cards, with the restric¬ 
tion that the only allowed operations are to look at the values of the top two cards, to 
exchange the top two cards, and to move the top card to the bottom of the deck. 

2.1.15 Expensive exchange. A clerk at a shipping company is charged with the task of 
rearranging a number of large crates in order of the time they are to be shipped out. 
Thus, the cost of compares is very low (just look at the labels) relative to the cost of ex¬ 
changes (move the crates). The warehouse is nearly full—there is extra space sufficient 
to hold any one of the crates, but not two. What sorting method should the clerk use? 

2.1.16 Certification. Write a checkO method that calls sort() for a given array and 
returns true if sort() puts the array in order and leaves the same set of objects in the 
array as were there initially, f al se otherwise. Do not assume that so rt O is restricted to 
move data only with exch(). You may use Arrays. sort() and assume that it is correct. 

2.1.17 Animation. Add code to Insertion and Selection to make them draw the 
array contents as vertical bars like the visual traces in this section, redrawing the bars 
after each pass, to produce an animated effect, ending in a “sorted” picture where the 
bars appear in order of their height. Hint: Use a client like the one in the text that gener¬ 
ates random Doubl e values, insert calls to show() as appropriate in the sort code, and 
implement a show() method that clears the canvas and draws the bars. 

2.1.18 Visual trace. Modify your solution to the previous exercise to make Insertion 
and Sel ecti on produce visual traces such as those depicted in this section. Hint: Judi¬ 
cious use of setYscal e () makes this problem easy. Extra credit: Add the code necessary 
to produce red and gray color accents such as those in our figures. 

2.1.19 Shellsort worst case. Construct an array of 100 elements containing the num¬ 
bers 1 through 100 for which shellsort, with the increments 1 4 13 40, uses as large a 
number of compares as you can find. 

2.1.20 Shellsort best case. What is the best case for shellsort? Justify your answer. 
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CREATIVE PROBLEMS (continued) 


2.1.21 Comparable transactions. Using our code for Date (page 247) as a model, ex¬ 
pand your implementation of Transaction (exercise 1 . 2 . 13 ) so that it implements 
Comparabl e, such that transactions are kept in order by amount. 

Solution: 

public class Transaction implements Comparable<Transaction> 

{ 

private final double amount; 

public int compareTo(Transaction that) 

{ 

if (this.amount > that.amount) return +1; 
if (this.amount < that.amount) return -1; 
return 0; 

} 

} 


2.1.22 Transaction sort test client. Write a class SortTransacti ons that consists of a 
static method mai n() that reads a sequence of transactions from standard input, sorts 
them, and prints the result on standard output (see exercise 1 . 3 . 17 ). 

Solution: 

public class SortTransactions 

{ 

public static Transaction[] readTransactions() 

{ // See Exercise 1.3.17 } 

public static void main(String[] args) 

{ 

Transaction[] transactions = readTransactions(); 

Shell.sort(transactions); 

for (Transaction t : transactions) 

StdOut.println(t); 

} 

} 
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EXPERIMENTS 

2.1.23 Deck sort. Ask a few friends to sort a deck of cards (see exercise 2 . 1 . 13 ). Ob¬ 
serve them carefully and write down the method(s) that they use. 

2.1.24 Insertion sort with sentinel. Develop an implementation of insertion sort that 
eliminates the j >0 test in the inner loop by first putting the smallest item into position. 
Use SortCompare to evaluate the effectiveness of doing so. Note: It is often possible to 
avoid an index-out-of-bounds test in this way—the element that enables the test to be 
eliminated is known as a sentinel. 

2.1.25 Insertion sort without exchanges. Develop an implementation of insertion sort 
that moves larger elements to the right one position with one array access per entry, 
rather than using exch(). Use SortCompare to evaluate the effectiveness of doing so. 

2.1.26 Primitive types. Develop a version of insertion sort that sorts arrays of int 
values and compare its performance with the implementation given in the text (which 
sorts Integer values and implicitly uses autoboxing and auto-unboxing to convert). 

2.1.27 Shellsort is subquadratic. Use SortCompare to compare shellsort with insertion 
sort and selection sort on your computer. Use array sizes that are increasing powers of 
2 , starting at 128. 

2.1.28 Equal keys. Formulate and validate hypotheses about the running time of in¬ 
sertion sort and selection sort for arrays that contain just two key values, assuming that 
the values are equally likely to occur. 

2.1.29 Shellsort increments. Run experiments to compare the increment sequence in 
algorithm 2.3 with the sequence 1, 5, 19, 41, 109, 209, 505, 929, 2161, 3905, 8929, 
16001, 36289, 64769, 146305, 260609 (which is formed by merging together the se¬ 
quences 9-4^— 9 . 2 * +1 and A k ~3-2 k + 1). See exercise 2 . 1 . 11 . 

2.1.30 Geometric increments. Run experiments to determine a value of t that leads to 
the lowest running time of shellsort for random arrays for the increment sequence 1 , 
[t\, \t 2 \, LDJ.LDJ,... for N= 10 6 . Give the values of t and the increment sequences for 
the best three values that you find. 
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EXPERIMENTS (continued) 


The following exercises describe various clients for helping to evaluate sorting methods. They 
are intended as starting points for helping to understand performance properties, using ran¬ 
dom data. In all of them, use timeO, as in SortCompare, 50 that you can get more accurate 
results by specifying more trials in the second command-line argument. We refer back to these 
exercises in later sections when evaluating more sophisticated methods. 

2.1.31 Doubling test. Write a client that performs a doubling test for sort algorithms. 
Start at N equal to 1000, and print N, the predicted number of seconds, the actual num¬ 
ber of seconds, and the ratio as N doubles. Use your program to validate that insertion 
sort and selection sort are quadratic for random inputs, and formulate and test a hy¬ 
pothesis for shellsort. 

2.1.32 Plot running times. Write a client that uses StdDrawto plotthe average running 
times of the algorithm for random inputs and various values of the array size. You may 
add one or two more command-line arguments. Strive to design a useful tool. 

2.1.33 Distribution. Write a client that enters into an infinite loop running so rt () on 
arrays of the size given as the third command-line argument, measures the time taken 
for each run, and uses StdDraw to plot the average running times. A picture of the dis¬ 
tribution of the running times should emerge. 

2.1.34 Corner cases. Write a client that runs sort() on difficult or pathological cases 
that might turn up in practical applications. Examples include arrays that are already 
in order, arrays in reverse order, arrays where all keys are the same, arrays consisting of 
only two distinct values, and arrays of size 0 or 1 . 

2.1.35 Nonuniform distributions. Write a client that generates test data by randomly 
ordering objects using other distributions than uniform, including the following: 

■ Gaussian 

■ Poisson 

■ Geometric 

■ Discrete (see exercise 2 . 1.28 for a special case) 

Develop and test hypotheses about the effect of such input on the performance of the 
algorithms in this section. 
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2.1.36 Nonuniform data. Write a client that generates test data that is not uniform, 
including the following: 

■ Half the data is Os, half Is. 

■ Half the data is Os, half the remainder is Is, half the remainder is 2s, and so forth. 

■ Half the data is Os, half random i nt values. 

Develop and test hypotheses about the effect of such input on the performance of the 
algorithms in this section. 

2.1.37 Partially sorted. Write a client that generates partially sorted arrays, including 
the following: 

■ 95 percent sorted, last percent random values 

■ All entries within 10 positions of their final place in the array 

■ Sorted except for 5 percent of the entries randomly dispersed throughout the 
array 

Develop and test hypotheses about the effect of such input on the performance of the 
algorithms in this section. 

2.1.38 Various types of items. Write a client that generates arrays of items of various 
types with random key values, including the following: 

■ Stri ng key (at least ten characters), one doubl e value 

■ doubl e key, ten Stri ng values (all at least ten characters) 

■ i nt key, one i nt [20] value 

Develop and test hypotheses about the effect of such input on the performance of the 
algorithms in this section. 
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2.2 MERGESORT 



the algorithms that we consider in this section are based on a simple operation 
known as merging: combining two ordered arrays to make one larger ordered array. 
This operation immediately leads to a simple recursive sort method known as merge- 
sort: to sort an array, divide it into two halves, sort the two halves (recursively), and 
then merge the results. As you will see, one of mergesort’s most attractive properties is 
that it guarantees to sort any array of N items in time proportional to Nlog N. Its prime 
disadvantage is that it uses extra space proportional to N. 
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Mergesort overview 


Abstract in-place merge The straightforward approach to implementing merg¬ 
ing is to design a method that merges two disjoint ordered arrays of Comparabl e ob¬ 
jects into a third array. This strategy is easy to implement: create an output array of the 
requisite size and then choose successively the smallest remaining item from the two 
input arrays to be the next item added to the output array. 

However, when we mergesort a large array, we are doing a huge number of merges, 
so the cost of creating a new array to hold the output every time that we do a merge is 
problematic. It would be much more desirable to have an in-place method so that we 
could sort the first half of the array in place, then sort the second half of the array in 
place, then do the merge of the two halves by moving the items around within the ar¬ 
ray, without using a significant amount of other extra space. It is worthwhile to pause 
momentarily to consider how you might do that. At first blush, this problem seems to 
be one that must be simple to solve, but solutions that are known are quite complicated, 
especially by comparison to alternatives that use extra space. 

Still, the abstraction of an in-place merge is useful. Accordingly, we use the method 
signature merge (a, lo, mid, hi ) to specify a merge method that puts the result of 
merging the subarrays a[lo. .mid] with a[mid+l. .hi] into a single ordered array, 
leaving the result in a[lo. .hi]. The code on the next page implements this merge 
method in just a few lines by copying everything to an auxiliary array and then merging 
back to the original. Another approach is described in exercise 2.2.10. 
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Abstract in-place merge 


public static void merge(Comparable[] a, int lo, int mid, int hi) 

{ // Merge a[lo..mid] with a[mid+l..hi]. 

int i = lo, j = mid+1; 


for (int k = lo; k <= hi; k++) // Copy a[lo..hi] to aux[lo..hi]. 

aux[k] = a[k]; 


} 


for (int k = lo; k <= hi 
if (i > mid) 

else if (j > hi ) 
else if (less(aux[j], 
el se 


; k++) // Merge back to a[lo..hi]. 

a[k] = aux[j++]; 
a[k] = aux[i++]; 
aux[i])) a[k] = aux[j++]; 

a[k] = aux[i++]; 


This method merges by first copying into the auxiliary array aux [] then merging back to a []. In the 
merge (the second for loop), there are four conditions: left half exhausted (take from the right), right 
half exhausted (take from the left), current key on right less than current key on left (take from the 
right), and current key on right greater than or equal to current key on left (take from the left). 
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Top-down mergesort algorithm 2.4 is a recur¬ 
sive mergesort implementation based on this abstract in- 
place merge. It is one of the best-known examples of the 
utility of the divide-and-conquer paradigm for efficient 
algorithm design. This recursive code is the basis for an 
inductive proof that the algorithm sorts the array: if it 
sorts the two subarrays, it sorts the whole array, by merg¬ 
ing together the subarrays. 

To understand mergesort, it is worthwhile to consider 
carefully the dynamics of the method calls, shown in the 
trace at right. To sort a [0 . . 15] , the sort() method calls 
itself to sort a[0. .7] then calls itself to sort a[0. .3] 
and a [ 0 . . 1 ] before finally doing the first merge of a [ 0 ] 
with a[l] after calling itself to sort a[ 0 ] and then a[l] 
(for brevity, we omit the calls for the base-case 1 -entry 
sorts in the trace). Then the next merge is a [2] with a [3] 
and then a[0. .1] with a[2. .3] and so forth. From this 
trace, we see that the sort code simply provides an orga¬ 
nized way to sequence the calls to the mergeO method. 
This insight will be useful later in this section. 

The recursive code also provides us with the basis for 
analyzing mergesort’s running time. Because mergesort 
is a prototype of the divide-and-conquer algorithm de¬ 
sign paradigm, we will consider this analysis in detail. 


sort 

left half 


sort 

right half 


merge 

results 


sort(a, 0, 15) 
sort(a, 0, 7) 
sort(a, 0, 3) 
sort(a, 0, 1) 

merge(a, 0, 0, 1) 
sort(a, 2, 3) 

merge(a, 2, 2, 3) 
merge(a, 0, 1, 3) 
sort(a, 4, 7) 
sort(a, 4, 5) 

merge(a, 4, 4, 5) 
sort(a, 6, 7) 

merge(a, 6, 6, 7) 
merge(a, 4, 5, 7) 
merge(a, 0, 3, 7) 
sort(a, 8, 15) 
sort(a, 8, 11) 
sort(a, 8, 9) 

merge(a, 8, 8, 9) 
sort(a, 10, 11) 

merge(a, 10, 10, 11) 
merge(a, 8, 9, 11) 
sort(a, 12, 15) 
sort(a, 12, 13) 

merge(a, 12, 12, 13) 
sort(a, 14, 15) 

merge(a, 14, 14,15) 
merge(a, 12, 13, 15) 
merge(a, 8, 11, 15) 
merge(a, 0, 7, 15) 


Top-down mergesort call trace 


Proposition F. Top-down mergesort uses between Vi Nig N and Nig N compares to 
sort any array of length N. 

Proof: Let C(N) be the number of compares needed to sort an array of length N. 
We have C( 0) = C( 1 ) = 0 and for N > 0 we can write a recurrence relationship that 
directly mirrors the recursive sortO method to establish an upper bound: 

C(N) < C(|AT/2j) + C([N/2]) + N 

The first term on the right is the number of compares to sort the left half of the ar¬ 
ray, the second term is the number of compares to sort the right half, and the third 
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ALGORITHM 2.4 Top-down mergesort 


public class Merge 

{ 

private static Comparable[] aux; // auxiliary array for merges 

public static void sort(Comparable[] a) 

{ 

aux = new Comparable[a.length]; // Allocate space just once. 

sort(a, 0, a.length - 1); 

} 

private static void sort(Comparable[] a, int lo, int hi) 

{ // Sort a[lo..hi]. 

if (hi <= lo) return; 

int mid = lo + (hi - lo)/2; 

sort(a, lo, mid); // Sort left half. 

sort(a, mid+1, hi); // Sort right half. 

merge(a, lo, mid, hi); // Merge results (code on page 271). 

} 

} 


To sort a subarray a [1 o.. hi ] we divide it into two parts: a [1 o. .mid] and a [mi d+1. .hi], sort them 
independently (via recursive calls), and merge the resulting ordered subarrays to produce the result. 
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term is the number of compares for the merge. The lower bound 
C(N) > C(|N/2j) + C(|\ZV/ 2 ]) + [N/2\ 

follows because the number of compares for the merge is at least | _N/2\ . 

We derive an exact solution to the recurrence when equality holds and N is a 
power of 2 (say N= 2"). First, since [N/2\ = \N/2~\ = 2"~ 1 , we have 

C( 2") = 2 C( 2 " -1 ) + 2 " 

Dividing both sides by 2” gives 

C{2 n )/2 n = C(2' !_ 1 )/2 " _1 + 1 

Applying the same equation to the first term on the right, we have 

C(2")/2' 1 = C( 2 n - 2 )/2 n ~ 2 +1 + 1 
Repeating the previous step n — 1 additional times gives 

C(2")/2" =C(2°)/2° + n 

which, after multiplying both sides by 2 ", leaves us with the solution 

C(N) = C(2") = n2 n =NlgN 

Exact solutions for general N are more complicated, but it is not difficult to apply 
the same argument to the inequalities describing the bounds on the number of 
compares to prove the stated result for all values of N. This proof is valid no matter 
what the input values are and no matter in what order they appear. 


Another way to understand proposition f is to examine the tree drawn below, where 
each node depicts a subarray for which sort() does a mergeO. The tree has precisely 
n levels. For k from 0 to n — 1, the Arth level from the top depicts 2 k subarrays, each of 
length 2" , each of which thus requires at most 2 " compares for the merge. Thus we 
have 2 k ■ 2 n ~ k = 2" total cost for each of the n levels, for a total of n2 n = /Vlg/V. 


(a[0..7] 


( a[0..3] ) 

A 



(a[Q..l] ) (a[2..3] ) (a[4..5] ) (a[6..7] ) (a[8..9] ) (a [10 ..11]) (a [12. .13]) (a [14. ,15])-L 


Mergesort subarray dependence tree for N = 16 
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Proposition G. Top-down mergesort uses at most 6 NlgN array accesses to sort an 
array of length N. 

Proof: Each merge uses at most 6 N array accesses (2 N for the copy, 2 N for the 
move back, and at most 2 N for compares). The result follows from the same argu¬ 
ment as for proposition f. 


propositions f and G tell us that we can expect the time required by mergesort to be 
proportional to Nlog N. That fact brings us to a different level from the elementary 
methods in section 2.1 because it tells us that we can sort huge arrays using just a 
logarithmic factor more time than it takes to examine every entry. You can sort millions 
of items (or more) with mergesort, but not with insertion sort or selection sort. The 
primary drawback of mergesort is that it requires extra space proportional to N, for 
the auxiliary array for merging. If space is at a premium, we need to consider another 
method. On the other hand, we can cut the running time of mergesort substantially 
with some carefully considered modifications to the implementation. 

Use insertion sort for small subarrays. We can improve most recursive algorithms by 
handling small cases differently, because the recursion guarantees that the method will 
be used often for small cases, so improvements in handling them lead to improvements 
in the whole algorithm. In the case of sorting, we know that insertion sort (or selection 
sort) is simple and therefore likely to be faster than mergesort for tiny subarrays. As 
usual, a visual trace provides insight into the operation of mergesort. The visual trace 
on the facing page shows the operation of a mergesort implementation with a cutoff 
for small subarrays. Switching to insertion sort for small subarrays (length 15 or less, 
say) will improve the running time of a typical mergesort implementation by 10 to 15 
percent (see exercise 2 . 2 . 23 ). 

Test whether the array is already in order. We can reduce the running time to be 
linear for arrays that are already in order by adding a test to skip the call to merge() if 
a [mi d] is less than or equal to a [mi d+1]. With this change, we still do all the recursive 
calls, but the running time for any sorted subarray is linear (see exercise 2 . 2 . 8 ). 

Eliminate the copy to the auxiliary array. It is possible to eliminate the time (but not 
the space) taken to copy to the auxiliary array used for merging. To do so, we use two 
invocations of the sort method: one takes its input from the given array and puts the 
sorted output in the auxiliary array; the other takes its input from the auxiliary array 
and puts the sorted output in the given array. With this approach, in a bit of recursive 
trickery, we can arrange the recursive calls such that the computation switches the roles 
of the input array and the auxiliary array at each level (see exercise 2.2.11). 
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first subarray 

second subarray 

first merge 


first half sorted 


second half sorted 


result 



Visual trace of top-down mergesort with cutoff for small subarrays 
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It is appropriate to repeat here a point raised in chapter i that is easily forgotten and 
needs reemphasis. Locally, we treat each algorithm in this book as if it were critical in 
some application. Globally, we try to reach general conclusions about which approach 
to recommend. Our discussion of such improvements is not necessarily a recommen¬ 
dation to always implement them, rather a warning not to draw absolute conclusions 
about performance from initial implementations. When addressing a new problem, 
your best bet is to use the simplest implementation with which you are comfortable 
and then refine it if it becomes a bottleneck. Addressing improvements that decrease 
running time just by a constant factor may not otherwise be worthwhile. You need to 
test the effectiveness of specific improvements by running experiments, as we indicate 
in exercises throughout. 

In the case of mergesort, the three improvements just listed are simple to implement 
and are of interest when mergesort is the method of choice—for example, in situations 
discussed at the end of this chapter. 

Bottom-up mergesort The recursive implementation of mergesort is prototypi¬ 
cal of the divide-and-conquer algorithm design paradigm, where we solve a large prob¬ 
lem by dividing it into pieces, solving the subproblems, then using the solutions for the 
pieces to solve the whole problem. Even though we are thinking in terms of merging 
together two large subarrays, the fact is that most merges are merging together tiny 
subarrays. Another way to implement mergesort is to organize the merges so that we do 
all the merges of tiny subarrays on one pass, then do a second pass to merge those sub¬ 
arrays in pairs, and so forth, continuing until we 
do a merge that encompasses the whole array. This 
method requires even less code than the standard 
recursive implementation. We start by doing a pass 
of 1-by-l merges (considering individual items as 
subarrays of size 1), then a pass of 2-by-2 merges 
(merge subarrays of size 2 to make subarrays of 
size 4), then 4-by-4 merges, and so forth. The sec¬ 
ond subarray may be smaller than the first in the 
last merge on each pass (which is no problem for 
mergeO), but otherwise all merges involve subar¬ 
rays of equal size, doubling the sorted subarray size 
for the next pass. 



Visual trace of bottom-up mergesort 


www.it-ebooks.info 



278 


CHAPTER 2 ■ Sorting 


Bottom-up mergesort 


public class MergeBU 

{ 

private static Comparable[] aux; // auxiliary array for merges 

// See page 271 for mergeO code. 

public static void sort(Comparable[] a) 

{ // Do lg N passes of pairwise merges. 

int N = a.length; 

aux = new Comparable[N]; 

for (int sz =1; sz < N; sz = sz+sz) // sz: subarray size 

for (int lo =0; lo < N-sz; lo += sz+sz) // lo: subarray index 
merge(a, lo, lo+sz-1, Math.min(lo+sz+sz-l, N-l)); 

} 

} 


Bottom-up mergesort consists of a sequence of passes over the whole array, doing sz-by-sz merges, 
starting with sz equal to 1 and doubling sz on each pass. The final subarray is of size sz only when 
the array size is an even multiple of sz (otherwise it is less than sz). 
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Trace of merge results for bottom-up mergesort 
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Proposition H. Bottom-up mergesort uses between ViNlg N and NlgN compares 
and at most 6 Nlg N array accesses to sort an array of length N. 

Proof: The number of passes through the array is precisely |_lg N\ (that is precisely 
the value of n such that 2" ^N< 2" +1 ). For each pass, the number of array accesses 
is exactly 6 N and the number of compares is at most N and no less than N/2. 


when the array length is a power of 2 , top-down and bottom-up mergesort per¬ 
form precisely the same compares and array accesses, just in a different order. When the 
array length is not a power of 2 , the sequence of compares and array accesses for the two 
algorithms will be different (see exercise 2 . 2 . 5 ). 

A version of bottom-up mergesort is the method of choice for sorting data orga¬ 
nized in a linked list. Consider the list to be sorted sublists of size 1, then pass through to 
make sorted subarrays of size 2 linked together, then size 4, and so forth. This method 
rearranges the links to sort the list in place (without creating any new list nodes). 

Both the top-down and bottom-up approaches to implementing a divide-and- 
conquer algorithm are intuitive. The lesson that you can take from mergesort is this: 
Whenever you encounter an algorithm based on one of these approaches, it is worth 
considering the other. Do you want to solve the problem by breaking it up into smaller 
problems (and solving them recursively) as in Merge, so rt () or by building small solu¬ 
tions into larger ones as in MergeBU. sortO? 

The complexity of sorting One important reason to know about mergesort is 
that we use it as the basis for proving a fundamental result in the field of computational 
complexity that helps us understand the intrinsic difficulty of sorting. In general, com¬ 
putational complexity plays an important role in the design of algorithms, and this 
result in particular is directly relevant to the design of sorting algorithms, so we next 
consider it in detail. 

The first step in a study of complexity is to establish a model of computation. Gen¬ 
erally, researchers strive to understand the simplest model relevant to a problem. For 
sorting, we study the class of compare-based algorithms that make their decisions about 
items only on the basis of comparing keys. A compare-based algorithm can do an ar¬ 
bitrary amount of computation between compares, but cannot get any information 
about a key except by comparing it with another one. Because of our restriction to the 
Comparabl e API, all of the algorithms in this chapter are in this class (note that we are 
ignoring the cost of array accesses), as are many algorithms that we might imagine. In 
chapter 5 , we consider algorithms that are not restricted to Comparabl e items. 
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Proposition I. No compare-based sorting algorithm can guarantee to sort N items 
with fewer than lg(N!) ~ N Ig N compares. 

Proof: First, we assume that the keys are all distinct, since any algorithm must be 
able to sort such inputs. Now, we use a binary tree to describe the sequence of com¬ 
pares. Each node in the tree is either a leaf (ip ij 1- 2 ■ ■ ■ i NnJ that indicates that the 
sort is complete and has discovered that the original inputs were in the order 
a[i 0 ] , a [i x ], . . or an internal node n : jj that corresponds to a com¬ 

pare operation between a[i] and a[j], with a left subtree corresponding to the 
sequence of compares in the case that a[i ] is less than a[j], and a right subtree 
corresponding to what happens if a[i] is greater than a[j]. Each path from the 
root to a leaf corresponds to the sequence of compares that the algorithm uses to 
establish the ordering given in the leaf. For example, here is a compare tree for 
N= 3: 



We never explicitly construct such a tree—it is a mathematical device for describ¬ 
ing the compares used by any algorithm. 

The first key observation in the proof is that the tree must have at least Nl leaves 
because there are N\ different permutations of N distinct keys. If there are fewer 
than N\ leaves, then some permutation is missing from the leaves, and the algo¬ 
rithm would fail for that permutation. 

The number of internal nodes on a path from the root to a leaf in the tree is the 
number of compares used by the algorithm for some input. We are interested in the 
length of the longest such path in the tree (known as the tree height) since it mea¬ 
sures the worst-case number of compares used by the algorithm. Now, it is a basic 
combinatorial property of binary trees that a tree of height h has no more than 2 h 
leaves—the tree of height h with the maximum number of leaves is perfectly bal¬ 
anced, or complete. An example for h = 4 is diagrammed on the next page. 
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any other tree of 
height 4 (black) has 
fewer than 16 leaves 


Combining the previous two paragraphs, we have shown that any compare-based 
sorting algorithm corresponds to a compare tree of height h with 


N! ^ number of leaves ^ 2 h 



h 


no more than 2^ leaves 


The value of h is precisely the worst-case number of compares, so we can take the 
logarithm (base 2) of both sides of this equation and conclude that the number of 
compares used by any algorithm must be at least lg 2V!. The approximation lg N! ~ 

N lg N follows immediately from Stirling’s approximation to the factorial function 
(see page 185). 

This result serves as a guide for us to know, when designing a sorting algorithm, how 
well we can expect to do. For example, without such a result, one might set out to try 
to design a compare-based sorting algorithm that uses half as many compares as does 
mergesort, in the worst case. The lower bound in proposition i says that such an effort 
is futile —no such algorithm exists. It is an extremely strong statement that applies to any 
conceivable compare-based algorithm. 

proposition h asserts that the number of compares used by mergesort in the worst 
case is ~ Nig N. This result is an upper bound on the difficulty of the sorting problem 
in the sense that a better algorithm would have to guarantee to use a smaller number of 
compares, proposition i asserts that no sorting algorithm can guarantee to use fewer 
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than ~N lg N compares. It is a lower bound on the difficulty of the sorting problem in 
the sense that even the best possible algorithm must use at least that many compares in 
the worst case. Together, they imply: 

Proposition J. Mergesort is an asymptotically optimal compare-based sorting 
algorithm. 

Proof: Precisely, we mean by this statement that both the number of compares used 
by mergesort in the worst case and the minimum number of compares that any com¬ 
pare-based sorting algorithm can guarantee are ~N lg N. Propositions h and i es¬ 
tablish these facts. 


It is important to note that, like the model of computation, we need to precisely define 
what we mean by an optimal algorithm. For example, we might tighten the definition 
of optimality and insist that an optimal algorithm for sorting is one that uses precisely 
lg N! compares. We do not do so because we could not notice the difference between 
such an algorithm and (for example) mergesort for large N. Or, we might broaden the 
definition of optimality to include any sorting algorithm whose worst-case number of 
compares is within a constant factor of Nig N. We do not do so because we might very 
well notice the difference between such an algorithm and mergesort for large N. 

computational complexity may seem rather abstract, but fundamental re¬ 
search on the intrinsic difficulty of solving computational problems hardly needs jus¬ 
tification. Moreover, when it does apply, it is emphatically the case that computational 
complexity affects the development of good software. First, good upper bounds allow 
software engineers to provide performance guarantees; there are many documented 
instances where poor performance has been traced to someone using a quadratic sort 
instead of a linearithmic one. Second, good lower bounds spare us the effort of search¬ 
ing for performance improvements that are not attainable. 

But the optimality of mergesort is not the end of the story and should not be mis¬ 
used to indicate that we need not consider other methods for practical applications. 
That is not the case because the theory in this section has a number of limitations. For 
example: 

■ Mergesort is not optimal with respect to space usage. 

■ The worst case may not be likely in practice. 

■ Operations other than compares (such as array accesses) may be important. 

■ One can sort certain data without using any compares. 

Thus, we shall be considering several other sorting methods in this book. 
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| Q&A 


Q. Is mergesort faster than shellsort? 

A. In practice, their running times are within a small constant factor of one another 
(when shellsort is using a well-tested increment sequence like the one in algorithm 
2 . 3 ), so comparative performance depends on the implementations. 

% java SortCompare Merge Shell 100000 
For 100000 random Double values 

Merge is 1.2 times faster than Shell 

In theory, no one has been able to prove that shellsort is linearithmic for random data, 
so there remains the possibility that the asymptotic growth of the average-case perfor¬ 
mance of shellsort is higher. Such a gap has been proven for worst-case performance, 
but it is not relevant in practice. 

Q. Why not make the aux[] array local to merge ()? 

A. To avoid the overhead of creating an array for every merge, even the tiny ones. This 
cost would dominate the running time of mergesort (see exercise 2 . 2 . 26 ). A more 
proper solution (which we avoid in the text to reduce clutter in the code) is to make 
aux[] local to sort() and pass it as an argument to merge() (see exercise 2 . 2 . 9 ). 

Q. How does mergesort fare when there are duplicate values in the array? 

A. If all the items have the same value, the running time is linear (with the extra test to 
skip the merge when the array is sorted), but if there is more than one duplicate value, 
this performance gain is not necessarily realized. For example, suppose that the input 
array consists of N items with one value in odd positions and N items with another 
value in even positions. The running time is linearithmic for such an array (it satisfies 
the same recurrence as for items with distinct values), not linear. 
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EXERCISES 

2.2.1 Give a trace, in the style of the trace given at the beginning of this section, show¬ 
ing how the keys AEQSUYEINOST are merged with the abstract in-place 
merge() method. 

2.2.2 Give traces, in the style of the trace given with algorithm 2 . 4 , showing how the 
keys EASYQUESTION are sorted with top-down mergesort. 

2.2.3 Answer exercise 2 . 2.2 for bottom-up mergesort. 

2.2.4 Does the abstract in-place merge produce proper output if and only if the two 
input subarrays are in sorted order? Prove your answer, or provide a counterexample. 

2.2.5 Give the sequence of subarray sizes in the merges performed by both the top- 
down and the bottom-up mergesort algorithms, for N = 39. 

2.2.6 Write a program to compute the exact value of the number of array accesses used 
by top-down mergesort and by bottom-up mergesort. Use your program to plot the val¬ 
ues for N from 1 to 512, and to compare the exact values with the upper bound 61Vlg N. 

2.2.7 Show that the number of compares used by mergesort is monotonically increas¬ 
ing (C(N+ 1) > C(N) for all N > 0). 

2.2.8 Suppose that algorithm 2.4 is modified to skip the call on mergeO whenever 
a [mi d] <= a [mi d+1]. Prove that the number of compares used to mergesort a sorted 
array is linear. 

2.2.9 Use of a static array like aux [] is inadvisable in library software because multiple 
clients might use the class concurrently. Give an implementation of Merge that does not 
use a static array. Do not make aux[] local to mergeO (see the Q&A for this section). 
Hint: Pass the auxiliary array as an argument to the recursive sortQ. 
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CREATIVE PROBLEMS 

2.2.10 Faster merge. Implement a version of merge() that copies the second half of 
a [] to aux [] in decreasing order and then does the merge back to a []. This change al¬ 
lows you to remove the code to test that each of the halves has been exhausted from the 
inner loop. Note: The resulting sort is not stable (see page 341). 

2.2.11 Improvements. Implement the three improvements to mergesort that are de¬ 
scribed in the text on page 275: Add a cutoff for small subarrays, test whether the array is 
already in order, and avoid the copy by switching arguments in the recursive code. 

2.2.12 Sublinear extra space. Develop a merge implementation that reduces the extra 
space requirement to max(M, N/M), based on the following idea: Divide the array into 
N/M blocks of size M (for simplicity in this description, assume that N is a multiple 
of M). Then, (i) considering the blocks as items with their first key as the sort key, sort 
them using selection sort; and (ii) run through the array merging the first block with 
the second, then the second block with the third, and so forth. 

2.2.13 Lower bound for average case. Prove that the expected number of compares used 
by any compare-based sorting algorithm must be at least ~N lg N (assuming that all 
possible orderings of the input are equally likely). Hint. The expected number of com¬ 
pares is at least the external path length of the compare tree (the sum of the lengths of 
the paths from the root to all leaves), which is minimized when it is balanced. 

2.2.14 Merging sorted queues. Develop a static method that takes two queues of sorted 
items as arguments and returns a queue that results from merging the queues into 
sorted order. 

2.2.15 Bottom-up queue mergesort. Develop a bottom-up mergesort implementation 
based on the following approach: Given N items, create N queues, each containing one 
of the items. Create a queue of the N queues. Then repeatedly apply the merging opera¬ 
tion of exercise 2 . 2.14 to the first two queues and reinsert the merged queue at the end. 
Repeat until the queue of queues contains only one queue. 

2.2.16 Natural mergesort. Write a version of bottom-up mergesort that takes advan¬ 
tage of order in the array by proceeding as follows each time it needs to find two arrays 
to merge: find a sorted subarray (by incrementing a pointer until finding an entry that 
is smaller than its predecessor in the array), then find the next, then merge them. Ana¬ 
lyze the running time of this algorithm in terms of the array size and the number of 
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CREATIVE PROBLEMS (continued) 


maximal increasing sequences in the array. 

2.2.17 Linked-list sort. Implement a natural mergesort for linked lists. (This is the 
method of choice for sorting linked lists because it uses no extra space and is guaranteed 
to be linearithmic.) 

2.2.18 Shuffling a linked list. Develop and implement a divide-and-conquer algo¬ 
rithm that randomly shuffles a linked list in linearithmic time and logarithmic extra 
space. 

2.2.19 Inversions. Develop and implement a linearithmic algorithm for computing 
the number of inversions in a given array (the number of exchanges that would be 
performed by insertion sort for that array—see section 2 . 1 ). This quantity is related 
to the Kendall tau distance; see section 2 . 5 . 

2.2.20 Indirect sort. Develop and implement a version of mergesort that does not re¬ 
arrange the array, but returns an i nt [] array perm such that perm[i ] is the index of the 
i th smallest entry in the array. 

2.2.21 Triplicates. Given three lists of N names each, devise a linearithmic algorithm 
to determine if there is any name common to all three lists, and if so, return the first 
such name. 

2.2.22 3-way mergesort. Suppose instead of dividing in half at each step, you divide 
into thirds, sort each third, and combine using a 3-way merge. What is the order of 
growth of the overall running time of this algorithm? 
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EXPERIMENTS 


2.2.23 Improvements. Run empirical studies to evaluate the effectiveness of each of the 
three improvements to mergesort that are described in the text (see exercise 2.2.11). 
Also, compare the performance of the merge implementation given in the text with the 
merge described in exercise 2.2.10. In particular, empirically determine the best value 
of the parameter that decides when to switch to insertion sort for small subarrays. 

2.2.24 Sort- test improvement. Run empirical studies for large randomly ordered ar¬ 
rays to study the effectiveness of the modification described in exercise 2.2.8 for ran¬ 
dom data. In particular, develop a hypothesis about the average number of times the 
test (whether an array is sorted) succeeds, as a function of N (the original array size for 
the sort). 

2.2.25 Multiway mergesort. Develop a mergesort implementation based on the idea of 
doing k- way merges (rather than 2-way merges). Analyze your algorithm, develop a hy¬ 
pothesis regarding the best value of k, and run experiments to validate your hypothesis. 

2.2.26 Array creation. Use SortCompare to get a rough idea of the effect on perfor¬ 
mance on your machine of creating aux[] in merge() rather than in sort(). 

2.2.27 Subarray lengths. Run mergesort for large random arrays, and make an empiri¬ 
cal determination of the average length of the other subarray when the first subarray 
exhausts, as a function of N (the sum of the two subarray sizes for a given merge). 

2.2.28 Top -down versus bottom-up. Use SortCompare to compare top-down and bot¬ 
tom-up mergesort for N=10 3 , 10 4 , 10 5 , and 10 6 . 

2.2.29 Natural mergesort. Determine empirically the number of passes needed in a 
natural mergesort (see exercise 2.2.16) for random Long keys with N= 10 3 , 10 6 , and 
10 9 . Hint: You do not need to implement a sort (or even generate full 64-bit keys) to 
complete this exercise. 
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2.3 QUICKSORT 



the subject of this section is the sorting algorithm that is probably used more 
widely than any other, quicksort. Quicksort is popular because it is not difficult to 
implement, works well for a variety of different kinds of input data, and is substantially 
faster than any other sorting method in typical applications. The quicksort algorithm’s 
desirable features are that it is in-place (uses only a small auxiliary stack) and that 
it requires time proportional to A/log N on the average to sort an array of length N. 
None of the algorithms that we have so far considered combine these two properties. 
Furthermore, quicksort has a shorter inner loop than most other sorting algorithms, 
which means that it is fast in practice as well as in theory. Its primary drawback is that 
it is fragile in the sense that some care is involved in the implementation to be sure to 
avoid bad performance. Numerous examples of mistakes leading to quadratic perfor¬ 
mance in practice are documented in the literature. Fortunately, the lessons learned 
from these mistakes have led to various improvements to the algorithm that make it of 
even broader utility, as we shall see. 

The basic algorithm Quicksort is a divide-and-conquer method for sorting. It 
works by partitioning an array into two subarrays, then sorting the subarrays indepen¬ 
dently. Quicksort is complementary to mergesort: for mergesort, we break the array 
into two subarrays to be sorted and then combine the ordered subarrays to make the 
whole ordered array; for quicksort, we rearrange the array such that, when the two 
subarrays are sorted, the whole array is ordered. In the first instance, we do the two 
recursive calls before working on the whole array; in the second instance, we do the two 
recursive calls after working on the whole array. For mergesort, the array is divided in 
half; for quicksort, the position of the partition depends on the contents of the array. 


input QUICKSORTEXAMPLE 
shuffle K iPUIMQCXOS 



—— 7 partitioning element 

partition E C A I E K * LPUTMQRXOS 


not greater 


not less 


sort left ACEEIKLPUTMQR 

sort right ACEEIKLMOPQRSTUX 

result ACEEIKLMOPQRSTUX 
Quicksort overview 
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ALGORITHM 2.5 Quicksort 


public class Quick 

{ 

public static void sort(Comparable[] a) 

{ 

StdRandom.shuffle(a); // Eliminate dependence on input. 

sort(a, 0, a.length - 1); 

} 

private static void sort(Comparable[] a, int lo, int hi) 

{ 

if (hi <= lo) return; 

int j = partition(a, lo, hi); // Partition (see page 291). 
sort(a, lo, j-1); // Sort left part a[lo .. j-1]. 

sort(a, j+1, hi); // Sort right part a[j+l .. hi]. 

} 

} 


Quicksort is a recursive program that sorts a subarray a[lo. . .hi] by using a parti ti on () method 
that puts a[i ] into position and arranges the rest of the entries such that the recursive calls finish 
the sort. 
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The crux of the method is the partitioning process, which rearranges the array to 
make the following three conditions hold: 

■ The entry a [ j ] is in its final place in the array, for some j. 

■ No entry in a [ 1 o ] through a [ j -1] is greater than a [ j ]. 

■ No entry in a [j+ 1 ] through a[hi ] is less than a[j]. 

We achieve a complete sort by partitioning, then recursively applying the method. 

Because the partitioning process always fixes one item into its position, a formal 
proof by induction that the recursive method constitutes a proper sort is not difficult 
to develop: if the left subarray and the right subarray are both properly sorted, then the 
result array, made up of the left subarray (in order, with no entry larger than the par¬ 
titioning item), the partitioning item, and the right subarray (in order, with no entry 
smaller that the partitioning item), is in order, algorithm 2.5 is a recursive program 

that implements this idea. It is a randomized algorithm, be¬ 
cause it randomly shuffles the array before sorting it. Our 
reason for doing so is to be able to predict (and depend 
upon) its performance characteristics, as discussed below. 

To complete the implementation, we need to implement 
the partitioning method. We use the following general strat¬ 
egy: First, we arbitrarily choose a [1 o] to be the partitioning 
item —the one that will go into its final position. Next, we 
scan from the left end of the array until we find an entry 
greater than (or equal to) the partitioning item, and we scan 
from the right end of the array until we find an entry less 
than (or equal to) the partitioning item. The two items that 
stopped the scans are out of place in the final partitioned array, so we exchange them. 
Continuing in this way, we ensure that no array entries to the left of the left index i are 
greater than the partitioning item, and no array entries to the right of the right index j 
are less than the partitioning item. When the scan indices cross, all that we need to do to 
complete the partitioning process is to exchange the partitioning item a [To] with the 
rightmost entry of the left subarray (a [ j ]) and return its index j. 

There are several subtle issues with respect to implementing quicksort that are re¬ 
flected in this code and worthy of mention, because each either can lead to incorrect 
code or can significantly impact performance. Next, we discuss several of these is¬ 
sues. Later in this section, we will consider three important higher-level algorithmic 
improvements. 



Quicksort partitioning overview 
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Quicksort partitioning 


private static int partition(Comparable[] a, int lo, int hi) 

{ // Partition into a[lo..i-l], a[i], a[i+l..hi]. 

int i = lo, j = hi+1; // left and right scan indices 

Comparable v = a[lo]; // partitioning item 

while (true) 

{ // Scan right, scan left, check for scan complete, and exchange. 

while (less(a[++i], v)) if (i == hi) break; 
while (less(v, a[ — j])) if (j == lo) break; 
if (i >= j) break; 
exch(a, i, j); 

} 

exch(a, lo, j); // Put v = a[j] into position 

return j; // with a[lo..j-l] <= a[j] <= a[j+l..hi]. 

} 


This code partitions on the item v in a [1 o]. The main loop exits when the scan indices i and j cross. 
Within the loop, we increment i while a [i ] is less than v and decrement j while a [ j ] is greater than 
v, then do an exchange to maintain the invariant property that no entries to the left of i are greater 
than v and no entries to the right of j are smaller than v. Once the indices meet, we complete the 
partitioning by exchanging a [1 o] with a [ j ] (thus leaving the partitioning value in a [ j ]). 
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Partitioning in place. If we use an extra array, partitioning is easy to implement, but 
not so much easier that it is worth the extra cost of copying the partitioned version back 
into the original. A novice lava programmer might even create a new spare array within 
the recursive method, for each partition, which would drastically slow down the sort. 

Staying in bounds. If the smallest item or the largest item in the array is the partition¬ 
ing item, we have to take care that the pointers do not run off the left or right ends of 
the array, respectively. Our parti tion() implementation has explicit tests to guard 
against this circumstance. The test (j == 1 o) is redundant, since the partitioning item 
is at a [To] and not less than itself. With a similar technique on the right it is not dif¬ 
ficult to eliminate both tests (see exercise 2 . 3 . 17 ). 

Preserving randomness. The random shuffle puts the array in random order. Since it 
treats all items in the subarrays uniformly, algorithm 2.5 has the property that its two 
subarrays are also in random order. This fact is crucial to the predictability of the algo¬ 
rithm’s running time. An alternate way to preserve randomness is to choose a random 
item for partitioning within parti ti on (). 

Terminating the loop. Experienced programmers know to take special care to ensure 
that any loop must always terminate, and the partitioning loop for quicksort is no ex¬ 
ception. Properly testing whether the pointers have crossed is a bit trickier than it might 
seem at first glance. A common error is to fail to take into account that the array might 
contain other items with the same key value as the partitioning item. 

Handling items with keys equal to the partitioning item’s key. It is best to stop the 
left scan for items with keys greater than or equal to the partitioning item’s key and 
the right scan for items with key less than or equal to the partitioning item’s key, as in 
algorithm 2 . 5 . Even though this policy might seem to create unnecessary exchanges 
involving items with keys equal to the partitioning item’s key, it is crucial to avoiding 
quadratic running time in certain typical applications (see exercise 2 . 3 . 11 ). Later, we 
discuss a better strategy for the case when the array contains a large number of items 
with equal keys. 

Terminating the recursion. Experienced programmers also know to take special care 
to ensure that any recursive method must always terminate, and quicksort is again no 
exception. For instance, a common mistake in implementing quicksort involves not 
ensuring that one item is always put into position, then falling into an infinite recursive 
loop when the partitioning item happens to be the largest or smallest item in the array. 
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Performance characteristics Quicksort has been subjected to very thorough 
mathematical analysis, so that we can make precise statements about its performance. 
The analysis has been validated through extensive empirical experience, and is a useful 
tool in tuning the algorithm for optimum performance. 

The inner loop of quicksort (in the partitioning method) increments an index and 
compares an array entry against a fixed value. This simplicity is one factor that makes 
quicksort quick: it is hard to envision a shorter inner loop in a sorting algorithm. For 
example, mergesort and shellshort are typically slower than quicksort because they also 
do data movement within their inner loops. 

The second factor that makes quicksort quick is that it uses few compares. Ulti¬ 
mately, the efficiency of the sort depends on how well the partitioning divides the array, 
which in turn depends on the value of the partitioning item’s key. Partitioning divides 
a large randomly ordered array into two smaller randomly ordered subarrays, but the 
actual split is equally likely (for distinct keys) to be anywhere in the array. Next, we 
consider the analysis of the algorithm, which allows us to see how this choice compares 
to the ideal choice. 

The best case for quicksort is when each partitioning stage divides the array exactly 
in half. This circumstance would make the number of compares used by quicksort 
satisfy the divide-and-conquer recurrence C N = 2C N/2 + N. The 2 C N/2 term covers the 
cost of sorting the two subarrays; the N is the cost of examining each entry, using one 
partitioning index or the other. As in the proof of proposition f for mergesort, we 
know that this recurrence has the solution C N ~ Nig N. Although things do not always 
go this well, it is true that the partition falls in the middle on the average. Taking into 
account the precise probability of each partition position makes the recurrence more 
complicated and more difficult to solve, but the final result is similar. The proof of 
this result is the basis for our confidence in quicksort. If you are not mathematically 
inclined, you may wish to skip (and trust) it; if you are mathematically inclined, you 
may find it intriguing. 


Proposition K. Quicksort uses ~ 2Nln N compares (and one-sixth that many ex¬ 
changes) on the average to sort an array of length N with distinct keys. 

Proof: Let C N be the average number of compares needed to sort N items with 
distinct values. We have C 0 = C 1 = 0 and for N> 1 we can write a recurrence relation¬ 
ship that directly mirrors the recursive program: 
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C n -N + 1 + (C 0 + Cj +...+ C N _ 2 + C N _ X ) /N + (C N _ l + C N _ 2 +...+ C 0 )/JV 

The first term is the cost of partitioning (always N + 1), the second term is the 
average cost of sorting the left subarray (which is equally likely to be any size from 
0 to N — 1), and the third term is the average cost for the right subarray (which is 
the same as for the left subarray). Multiplying by N and collecting terms transforms 
this equation to 

NC n =N(N + 1) + 2(C 0 + C 1 + .. .+C N _ 2 +C N _j) 

Subtracting this from the same equation for N — 1 gives 

NC n ~ (N — 1)0^!= 2N + 2C n _ 1 

Rearranging terms and dividing by N{N + 1) leaves 

C n /{N + 1) = C^/N + 2/(N + 1) 

which telescopes to give the result 

C N ~2{N+ l)(l/3 + 1/4 + ... + 1 /{N+ 1) ) 

The parenthesized quantity is the discrete estimate of the area under the curve 2/x 
from 3 to N, + 1 and C N ~ 2NlnNby integration. Note that 2iVln N ~ 1,39/Vlg 
N, so the average number of compares is only about 39 percent higher than in the 
best case. 

A similar (but much more complicated) analysis is needed to establish the stated 
result for exchanges. 


When keys may not be distinct, as is typical in practical applications, precise anaysis is 
considerably more complicated, but it is not difficult to show that the average number 
of compares is no greater than than C N , even when duplicate keys may be present (on 
page 296, we will look at a way to improve quicksort in this case). 

Despite its many assets, the basic quicksort program has one potential liability: it 
can be extremely inefficient if the partitions are unbalanced. For example, it could be 
the case that the first partition is on the smallest item, the second partition on the next 
smallest item, and so forth, so that the program will remove just one item for each call, 
leading to an excessive number of partitions of large subarrays. Avoiding this situation 
is the primary reason that we randomly shuffle the array before using quicksort. This 
action makes it so unlikely that bad partitions will happen consistently that we need not 
worry about the possibility. 
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Proposition L. Quicksort uses ~ N 2 /2 compares in the worst case, but random 
shuffling protects against this case. 

Proof: By the argument just given, the number of compares used when one of the 
subarrays is empty for every partition is 

N+(N-l) + (N-2)+ . .. + 2+ l = (N+ l)N/2 

This behavior means not only that the time required will be quadratic but also that 
the space required to handle the recursion will be linear, which is unacceptable for 
large arrays. But (with quite a bit more work) it is possible to extend the analysis 
that we did for the average to find that the standard deviation of the number of 
compares is about .65 N, so the running time tends to the average as N grows and is 
unlikely to be far from the average. For example, even the rough estimate provided 
by Chebyshev’s inequality says that the probability that the running time is more 
than ten times the average for an array with a million elements is less than .00001 
(and the true probability is far smaller). The probability that the running time for 
a large array is close to quadratic is so remote that we can safely ignore the pos¬ 
sibility (see exercise 2 . 3 . 10 ). For example, the probability that quicksort will use 
as many compares as insertion sort or selection sort when sorting a large array on 
your computer is much less than the probability that your computer will be struck 
by lightning during the sort! 


in summary, you can be sure that the running time of algorithm 2.5 will be within 
a constant factor of 1 .39 N lg N whenever it is used to sort N items. The same is true 
of mergesort, but quicksort is typically faster because (even though it does 39 per¬ 
cent more compares) it does much less data movement. This mathematical assurance is 
probabilistic, but you can certainly rely upon it. 

Algorithmic improvements Quicksort was invented in 1960 by C. A. R. Hoare, 
and many people have studied and refined it since that time. It is tempting to try to 
develop ways to improve quicksort: a faster sorting algorithm is computer science’s 
“better mousetrap,” and quicksort is a venerable method that seems to invite tinkering. 
Almost from the moment Hoare first published the algorithm, people began proposing 
ways to improve the algorithm. Not all of these ideas are fully successful, because the al¬ 
gorithm is so well-balanced that the effects of improvements can be more than offset by 
unexpected side effects, but a few of them, which we now consider, are quite effective. 
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If your sort code is to be used a great many times or to sort a huge array (or, in par¬ 
ticular, if it is to be used as a library sort that will be used to sort arrays of unknown 
characteristics), then it is worthwhile to consider the improvements that are discussed 
in the next few paragraphs. As noted, you need to run experiments to determine the 
effectiveness of these improvements and to determine the best choice of parameters for 
your implementation. Typically, improvements of 20 to 30 percent are available. 

Cutoff to insertion sort. As with most recursive algorithms, an easy way to improve 
the performance of quicksort is based on the following two observations: 

■ Quicksort is slower than insertion sort for tiny subarrays. 

■ Being recursive, quicksort’s sort() is certain to call itself for tiny subarrays. 
Accordingly, it pays to switch to insertion sort for tiny subarrays. A simple change to 
algorithm 2.5 accomplishes this improvement: replace the statement 

if (hi <= lo) return; 

in sortO with a statement that invokes insertion sort for small subarrays: 

if (hi <= lo + M) { Insertion.sort(a, lo, hi); return; } 

The optimum value of the cutoff M is system-dependent, but any value between 5 and 
15 is likely to work well in most situations (see exercise 2 . 3 . 25 ). 

Median-of-three partitioning. A second easy way to improve the performance of 
quicksort is to use the median of a small sample of items taken from the subarray as the 
partitioning item. Doing so will give a slightly better partition, but at the cost of com¬ 
puting the median. It turns out that most of the available improvement comes from 
choosing a sample of size 3 and then partitioning on the middle item (see exercises 
2 . 3.18 and 2 . 3 . 19 ). As a bonus, we can use the sample items as sentinels at the ends of 
the array and remove both array bounds tests in parti ti on () ■ 

Entropy-optimal sorting. Arrays with large numbers of duplicate keys arise fre¬ 
quently in applications. For example, we might wish to sort a large personnel file 
by year of birth, or perhaps to separate females from males. In such situations, the 
quicksort implementation that we have considered has acceptable performance, 
but it can be substantially improved. For example, a subarray that consists solely of 
items that are equal (just one key value) does not need to be processed further, but 
our implementation keeps partitioning down to small subarrays. In a situation where 
there are large numbers of duplicate keys in the input array, the recursive nature of 
quicksort ensures that subarrays consisting solely of items with keys that are equal will 
occur often. There is potential for significant improvement, from the linearithmic-time 
performance of the implementations seen so far to linear-time performance. 
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Quicksort with median-of-3 partitioning and cutoff for small subarrays 
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One straightforward idea is to partition the array into three parts, one each for 
items with keys smaller than, equal to, and larger than the partitioning item’s key. 
Accomplishing this partitioning is more complicated than the 2-way partitioning that 
we have been using, and various different methods have been suggested for the task. 
It was a classical programming exercise popularized by E. W. Dijkstra as the Dutch 
National Flag problem, because it is like sorting an array with three possible key values, 
which might correspond to the three colors on the flag. 

Dijkstra’s solution to this problem leads to the remarkably simple partition code 
shown on the next page. It is based on a single left-to-right pass through the array that 
maintains a pointer It such that a [To. . 1 t- 1 ] is less than v, a pointer gt such that 
a [gt+ 1 , hi ] is greater than v, and a pointer i such that a [ 11 . . i - 1 ] are equal to v and 
a [i. . gt] are not yet examined. Starting with i equal to 1 o, we process a [i ] using the 
3-way comparison given us by the Comparable interface (instead of using lessO) to 
directly handle the three possible cases: 

■ a [i ] less than v: exchange a [ 11 ] with a [i ] and increment both 11 and i 

■ a[i ] greater than v: exchange a[i ] with a[gt] and decrement gt 

■ a[i ] equal to v: increment i 

Each of these operations both maintains the invariant and decreases the value of 
gt-i (so that the loop terminates). Furthermore, every item encountered leads to an 
exchange except for those items with keys equal to the partitioning item’s key. 

Though this code was developed not long after quicksort in the 1970s, it fell out of 
favor because it uses many more exchanges 
than the standard 2 -way partitioning method 
for the common case when the number of 
duplicate keys in the array is not high. In the 
1990s J. Bentley and D. Mcllroy developed a 
clever implementation that overcomes this 
problem (see exercise 2 . 3 . 22 ), and observed 
that 3-way partitioning makes quicksort 
asymptotically faster than mergesort and 
other methods in practical situations 
involving large numbers of equal keys. Later, 

J. Bentley and R. Sedgewick developed a proof of this fact, which we discuss next. 

But we proved that mergesort is optimal. How have we defeated that lower bound? 
The answer to this question is that proposition i in section 2.2 addresses worst- 
case performance over all possible inputs, while now we are looking at worst-case 
performance with some information about the key values at hand. Mergesort does not 
guarantee optimal performance for any given distribution of duplicates in the input: 
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Quicksort with 3-way partitioning 


public class Quick3way 

{ 

private static void sort(Comparable[] a, int lo, int hi) 

{ // See page 289 for public sort() that calls this method. 

if (hi <= lo) return; 

int It = lo, i = lo+l, gt = hi; 

Comparable v = a[lo]; 
while (i <= gt) 

{ 

int cmp = a[i].compareTo(v); 
if (cmp < 0) exch(a, lt++, i++); 

else if (cmp > 0) exch(a, i, gt--); 
else i++; 

} // Now a[lo..lt-l] < v = a[lt..gt] < a[gt+l..hi]. 

sort(a, lo, It - 1); 
sort(a, gt + 1, hi); 

} 

} 


This sort code partitions to put keys equal to the partitioning element in place and thus does not have 
to include those keys in the subarrays for the recursive calls. It is far more efficient than the standard 
quicksort implementation for arrays with large numbers of duplicate keys (see text). 
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3-way partitioning trace (array contents after each loop iteration) 
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Visual trace of quicksort with 3-way partitioning 


for example, mergesort is linearithmic for a randomly ordered array that has only a 
constant number of distinct key values, but quicksort with 3-way partitioning is linear 
for such an array. Indeed, by examining the visual trace above, you can see that N times 
the number of key values is a conservative bound on the running time. 

The analysis that makes these notions precise takes the distribution of key values 
into account. Given N keys with k distinct key values, for each i from 1 to k define f to 
be frequency of occurrence of the i th key value and />, to b ef/ N, the probability that 
the zth key value is found when a random entry of the array is sampled. The Shannon 
entropy of the keys (a classic measure of their information content) is defined as 

H =~(Pi IgPi + P2 kP 2 + ■ ■ • + Pk l SPk ) 

Given any array of items to be sorted, we can calculate its entropy by counting the fre¬ 
quency of each key value. Remarkably, we can also derive from the entropy both a lower 
bound on the number of compares and an upper bound on the number of compares 
used by quicksort with 3-way partitioning. 

Proposition M. No compare-based sorting algorithm can guarantee to sort N items 
with fewer than NH — N compares, where H is the Shannon entropy, defined from 
the frequencies of key values. 

Proof sketch: This result follows from a (relatively easy) generalization of the low¬ 
er bound proof of proposition i in section 2 . 2 . 
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Proposition N. Quicksort with 3-way partitioning uses ~ (2ln 2) N H compares to 
sort N items, where H is the Shannon entropy, defined from the frequencies of key 
values. 

Proof sketch: This result follows from a (relatively difficult) generalization of the 
average-case analysis of quicksort in proposition k. As with distinct keys, this 
costs about 39 percent more than the optimum (but within a constant factor). 


Note that H = lg N when the keys are all distinct (all the probabilities are 1 IN), which 
is consistent with proposition i in section 2.2 and proposition k. The worst case 
for 3-way partitioning happens when the keys are distinct; when duplicate keys are 
present, it can do much better than mergesort. More important, these two properties 
together imply that quicksort with 3-way partitioning is entropy-optimal, in the sense 
that the average number of compares used by the best possible compare-based sorting 
algorithm and the average number of compares used by 3-way quicksort are within a 
constant factor of one another, for any given distribution of input key values. 

As with standard quicksort, the running time tends to the average as the array size 
grows, and large deviations from the average are extremely unlikely, so that you can 
depend on 3-way quicksort’s running time to be proportional to N times the entropy 
of the distribution of input key values. This property of the algorithm is important in 
practice because it reduces the time of the sort from linearithmic to linear for arrays with 
large numbers of duplicate keys. The order of the keys is immaterial, because the algo¬ 
rithm shuffles them to protect against the worst case. The distribution of keys defines 
the entropy and no compare-based algorithm can use fewer compares than defined by 
the entropy. This ability to adapt to duplicates in the input makes 3-way quicksort the 
algorithm of choice for a library sort—clients that sort arrays containing large numbers 
of duplicate keys are not unusual. 

A carefully tuned version of quicksort is likely to run significantly faster on most 
computers for most applications than will any other compare-based sorting method. 
Quicksort is widely used throughout today’s computational infrastructLire because 
the mathematical models that we have discussed suggest that it will outperform other 
methods in practical applications, and extensive experiments and experience over the 
past several decades have validated that conclusion. 

We will see in chapter 5 that this is not quite the end of the story in the development 
of sorting algorithms, because is it possible to develop algorithms that do not use 
compares at all! But a version of quicksort turns out to be best in that situation, as well. 
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| Q&A 

Q. Is there some way to just divide the array into two halves, rather than letting the 
partitioning element fall where it may? 

A. That is a question that stumped experts for over a decade. It is tantamount to find¬ 
ing the median key value in the array and then partitioning on that value. We discuss 
the problem of finding the median on page 346. It is possible to do so in linear time, but 
the cost of doing so with known algorithms (which are based on quicksort partition¬ 
ing!) far exceeds the 39 percent savings available from splitting the array into equal 
parts. 

Q. Randomly shuffling the array seems to take a significant fraction of the total time 
for the sort. Is doing so really worthwhile? 

A. Yes. It protects against the worst case and makes the running time predictable. Hoare 
proposed this approach when he presented the algorithm in 1960—it is a prototypical 
(and among the first) randomized algorithm. 

Q. Why all the focus on items with equal keys? 

A. The issue directly impacts performance in practical situations. It was overlooked by 
many for decades, with the result that some older implementations of quicksort take 
quadratic time for arrays with large numbers of items with equal keys, which certainly 
do arise in applications. Better implementations such as algorithm 2.5 take linearith- 
mic time for such arrays, but improving that to linear-time as in the entropy-optimal 
sort at the end of this section is worthwhile in many situations. 
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EXERCISES 

2 . 3.1 Show, in the style of the trace given with parti ti on (), how that method pati- 
tions the array EASYQUESTION. 

2 . 3.2 Show, in the style of the quicksort trace given in this section, how quicksort sorts 
the array EASYQUESTION (for the purposes of this exercise, ignore the 
initial shuffle). 

2 . 3.3 What is the maximum number of times during the execution of Qui ck. sort() 
that the largest item can be exchanged, for an array of length N ? 

2 . 3.4 Suppose that the initial random shuffle is omitted. Give six arrays of ten elements 
for which Qui ck. sortO uses the worst-case number of compares. 

2 . 3.5 Give a code fragment that sorts an array that is known to consist of items having 
just two distinct keys. 

2 . 3.6 Write a program to compute the exact value of C N , and compare the exact value 
with the approximation 2Nln N, for N = 100,1,000, and 10,000. 

2 . 3.7 Find the expected number of subarrays of size 0,1, and 2 when quicksort is used 
to sort an array of N items with distinct keys. If you are mathematically inclined, do the 
math; if not, run some experiments to develop hypotheses. 

2 . 3.8 About how many compares will Qui ck. sortO make when sorting an array of 
N items that are all equal? 

2 . 3.9 Explain what happens when Qui ck. sortO is run on an array having items with 
just two distinct keys, and then explain what happens when it is run on an array having 
just three distinct keys. 

2 . 3.10 Chebyshev’s inequality says that the probability that a random variable is more 
than k standard deviations away from the mean is less than 1 Ik 2 . For ;Y = 1 million, use 
Chebyshev’s inequality to bound the probability that the number of compares used by 
quicksort is more than 100 billion (.1 N 2 ). 

2 . 3 . 11 Suppose that we scan over items with keys equal to the partitioning item’s key 
instead of stopping the scans when we encounter them. Show that the running time 
of this version of quicksort is quadratic for all arrays with just a constant number of 
distinct keys. 
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EXERCISES (continued) 


2 . 3.12 Show, in the style of the trace given with the code, how the entropy-optimal sort 
first partitions the array BABABABACADABRA. 

2 . 3.13 What is the recursive depth of quicksort, in the best, worst, and average cases? 
This is the size of the stack that the system needs to keep track of the recursive calls. See 
exercise 2 . 3.20 for a way to guarantee that the recursive depth is logarithmic in the 
worst case. 

2 . 3.14 Prove that when running quicksort on an array with N distinct items, the prob¬ 
ability of comparing the z’th and j th largest items is 21 (j — i). Then use this result to 
prove proposition k. 
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CREATIVE PROBLEMS 


2 . 3.15 Nuts and bolts. (G. J. E. Rawlins) You have a mixed pile of N nuts and N bolts 
and need to quickly find the corresponding pairs of nuts and bolts. Each nut matches 
exactly one bolt, and each bolt matches exactly one nut. By fitting a nut and bolt to¬ 
gether, you can see which is bigger, but it is not possible to directly compare two nuts or 
two bolts. Give an efficient method for solving the problem. 

2 . 3.16 Best case. Write a program that produces a best-case array (with no duplicates) 
for sort() in algorithm 2 . 5 : an array of N items with distinct keys having the prop¬ 
erty that every partition will produce subarrays that differ in size by at most 1 (the same 
subarray sizes that would happen for an array of N equal keys). (For the purposes of this 
exercise, ignore the initial shuffle.) 

The following exercises describe variants of quicksort. Each of them calls for an implementa¬ 
tion, but naturally you will also want to use SortCompare for experiments to evaluate the 
effectiveness of each suggested modification. 

2 . 3.17 Sentinels. Modify the code in algorithm 2.5 to remove both bounds checks 
in the inner while loops. The test against the left end of the subarray is redundant since 
the partitioning item acts as a sentinel (v is never less than a [1 o]). To enable removal of 
the other test, put an item whose key is the largest in the whole array into a [1 ength-1] 
just after the shuffle. This item will never move (except possibly to be swapped with an 
item having the same key) and will serve as a sentinel in all subarrays involving the end 
of the array. Note: When sorting interior subarrays, the leftmost entry in the subarray 
to the right serves as a sentinel for the right end of the subarray. 

2 . 3.18 Median-of-3 partitioning. Add median-of-3 partitioning to quicksort, as de¬ 
scribed in the text (see page 296). Run doubling tests to determine the effectiveness of 
the change. 

2 . 3.19 Median-of-5 partitioning. Implement a quicksort based on partitioning on the 
median of a random sample of five items from the stibarray. Put the items of the sample 
at the appropriate ends of the array so that only the median participates in partitioning. 
Run doubling tests to determine the effectiveness of the change, in comparison both 
to the standard algorithm and to median-of-3 partitioning (see the previous exercise). 
Extra credit: Devise a median-of-5 algorithm that uses fewer than seven compares on 
any input. 
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CREATIVE PROBLEMS (continued) 


2 . 3.20 Nonrecursive quicksort. Implement a nonrecursive version of quicksort based 
on a main loop where a subarray is popped from a stack to be partitioned, and the re¬ 
sulting subarrays are pushed onto the stack. Note: Push the larger of the subarrays onto 
the stack first, which guarantees that the stack will have at most lg N entries. 

2 . 3.21 Lower bound for sorting with equal keys. Complete the first part of the proof 
of proposition m by following the logic in the proof of proposition i and using the 
observation that there are N\ / j\ Iff. .. .f k \ different ways to arrange keys with k different 
values, where the i th value appears with frequency/; (= Np t , in the notation of proposi¬ 
tion m), with/j-K .. +f k = N. 

2 . 3.22 Fast 3-way partitioning. (J. Bentley and D. Mcllroy) Implement an entropy- 
optimal sort based on keeping item's with equal keys at both the left and right ends 

of the subarray. Maintain indices p and q such that 
a[lo. .p-1] and a[q+l. .hi] are all equal to a[lo], 
an index i such that a [ p. . i -1] are all less than a [ 1 o ], 
and an index j such that a [ j +1. . q] are all greater than 
a [1 o]. Add to the inner partitioning loop code to swap 
a [i ] with a [p] (and increment p) if it is equal to v and 
to swap a [ j ] with a [q] (and decrement q) if it is equal 
to v before the usual comparisons of a[i] and a[j] 
with v. After the partitioning loop has terminated, add 
code to swap the items with equal keys into position. 
Note: This code complements the code given in the 
text, in the sense that it does extra swaps for keys equal to the partitioning item’s key, 
while the code in the text does extra swaps for keys that are not equal to the partitioning 
item’s key. 

2 . 3.23 Java system sort. Add to your implementation from exercise 2 . 3.22 code to 
use the Tukey ninther to compute the partitioning item—choose three sets of three 
items, take the median of each, then use the median of the three medians as the parti¬ 
tioning item. Also, add a cutoff to insertion sort for small subarrays. 

2 . 3.24 Samplesort. (W. Frazer and A. McKellar) Implement a quicksort based on us¬ 
ing a sample of size 2* — 1. First, sort the sample, then arrange to have the recursive 
routine partition on the median of the sample and to move the two halves of the rest of 
the sample to each subarray, such that they can be used in the subarrays, without having 
to be sorted again. This algorithm is called samplesort. 



lo j i hi 

Bentley-Mcllroy 3-way partitioning 
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EXPERIMENTS 


2 . 3.25 Cutoff to insertion sort. Implement quicksort with a cutoff to insertion sort 
for subarrays with less than M elements, and empirically determine the value of M for 
which quicksort runs fastest in your computing environment to sort random arrays 
of N doubles, for N = 10 3 , 10 4 , 10 5 , and 10 6 . Plot average running times for M from 0 
to 30 for each value of M. Note: You need to add a three-argument sort() method to 
algorithm 2.2 for sorting subarrays such that the call Inserti on. sort (a, 1 o, hi) 
sorts the subarray a [1 o. . hi ]. 

2 . 3.26 Subarray sizes. Write a program that plots a histogram of the subarray sizes left 
for insertion sort when you run quicksort for an array of size N with a cutoff for subar¬ 
rays of size less than M. Run your program for M=10, 20, and 50 and N = 10 5 . 

2 . 3.27 Ignore small subarrays. Run experiments to compare the following strategy for 
dealing with small subarrays with the approach described in exercise 2 . 3 . 25 : Simply 
ignore the small subarrays in quicksort, then run a single insertion sort after the quick¬ 
sort completes. Note: You may be able to estimate the size of your computer’s cache 
memory with these experiments, as the performance of this method is likely to degrade 
when the array does not fit in the cache. 

2 . 3.28 Recursion depth. Run empirical studies to determine the average recursive 
depth used by quicksort with cutoff for arrays of size M, when sorting arrays of N 
distinct elements, for M=10,20, and 50 and N = 10 3 , 10 4 , 10 5 , and 10 6 . 

2 . 3.29 Randomization. Run empirical studies to compare the effectiveness of 
the strategy of choosing a random partitioning item with the strategy of initially 
randomizing the array (as in the text). Use a cutoff for arrays of size M, and sort arrays 
of N distinct elements, for M=10,20, and 50 and N = 10 3 , 10 4 , 10 5 , and 10 6 . 

2 . 3.30 Corner cases. Test quicksort on large nonrandom arrays of the kind described 
in exercises 2 . 1.35 and 2 . 1.36 both with and without the initial random shuffle. How 
does shuffling affect its performance for these arrays? 

2 . 3.31 Histogram of running times. Write a program that takes command-line argu¬ 
ments N and T, does T trials of the experiment of running quicksort on an array of 
N random Double values, and plots a histogram of the observed running times. Run 
your program for N = 10 3 , 10 4 , 10 5 , and 10 6 , with T as large as you can afford to make 
the curves smooth. Your main challenge for this exercise is to appropriately scale the 
experimental results. 
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2.4 PRIORITY QUEUES 



many applications require that we process items having keys in order, but not nec¬ 
essarily in full sorted order and not necessarily all at once. Often, we collect a set of 
items, then process the one with the largest key, then perhaps collect more items, then 
process the one with the current largest key, and so forth. For example, you are likely to 
have a computer (or a cellphone) that is capable of running several applications at the 
same time. This effect is typically achieved by assigning a priority to events associated 
with applications, then always choosing to process next the highest-priority event. For 
example, most cellphones are likely to process an incoming call with higher priority 
than a game application. 

An appropriate data type in such an environment supports two operations: remove 
the maximum and insert. Such a data type is called a priority queue. Using priority 
queues is similar to using queues (remove the oldest) and stacks (remove the newest), 
but implementing them efficiently is more challenging. 

In this section, after a short discussion of elementary representations where one or 
both of the operations take linear time, we consider a classic priority-queue implemen¬ 
tation based on the binary heap data structure, where items are kept in an array, subject 
to certain ordering constraints that allow for efficient (logarithmic-time) implementa¬ 
tions of remove the maximum and insert. 

Some important applications of priority queues include simulation systems, where 
the keys correspond to event times, to be processed in chronological order; job schedul¬ 
ing, where the keys correspond to priorities indicating which tasks are to be performed 
first; and numerical computations, where the keys represent computational errors, in¬ 
dicating in which order we should deal with them. We consider in chapter 6 a detailed 
case study showing the use of priority queues in a particle-collision simulation. 

We can use any priority queue as the basis for a sorting algorithm by inserting a se¬ 
quence of items, then successively removing the smallest to get them out, in order. An 
important sorting algorithm known as heap sort also follows naturally from our heap- 
based priority-queue implementations. Later on in this book, we shall see how to use 
priority queues as building blocks for other algorithms. In chapter 4 , we shall see how 
priority queues are an appropriate abstraction for implementing several fundamental 
graph-searching algorithms; in chapter 5 , we shall develop a data-compression algo¬ 
rithm using methods from this section. These are but a few examples of the important 
role played by the priority queue as a tool in algorithm design. 
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API The priority queue is a prototypical abstract data type (see section 1 . 2 ): it rep¬ 
resents a set of values and operations on those values, and it provides a convenient ab¬ 
straction that allows us to separate application programs (clients) from various imple¬ 
mentations that we will consider in this section. As in section 1 . 2 , we precisely define 
the operations by specifying an applications programming interface (API) that provides 
the information needed by clients. Priority queues are characterized by the remove the 
maximum and insert operations, so we shall focus on them. We use the method names 
del Max () for remove the maximum and insertO for insert. By convention, we will 
compare keys only with a helper lessO method, as we have been doing for sorting. 
Thus, if items can have duplicate keys, maximum means any item with the largest key 
value. To complete the API, we also need to add constructors (like the ones we used for 
stacks and queues) and a test if empty operation. For flexibility, we use a generic imple¬ 
mentation with a parameterized type Key that implements the Comparable interface. 
This choice eliminates our distinction between items and keys and enables clearer and 
more compact descriptions of data structures and algorithms. For example, we refer to 
the “largest key” instead of the “largest item” or the “item with the largest key.” 

For convenience in client code, the API includes three constructors, which enable 
clients to build priority queues of an initial fixed size (perhaps initialized with a given 
array of keys). To clarify client code, we will use a separate class Mi nPQ whenever ap¬ 
propriate, which is the same as MaxPQ except that it has a del Mi n () method that deletes 
and returns an item with the smallest key in the queue. Any MaxPQ implementation is 
easily converted into a Mi nPQ implementation and vice versa, simply by reversing the 
sense of the comparison in 1 ess (). 

public class MaxPQ<Key extends Comparable<Key» 


MaxPQQ 


create a priority queue 


MaxPQ (int max) create a priority queue of initial capacity max 
MaxPQ(Key[] a) create a priority queue from the keys in a[] 


void insert(Key v) insert a key into the priority queue 


Key max() 


return the largest key 
return and remove the largest key 
is the priority queue empty? 
number of keys in the priority queue 


Key delMaxO 
boolean isEmptyO 


int sizeQ 


API for a generic priority queue 


www.it-ebooks.info 



310 


CHAPTER 2 ■ Sorting 


A priority-queue client. To appreciate the 
value of the priority-queue abstraction, con¬ 
sider the following problem: You have a huge 
input stream of N strings and associated inte¬ 
ger values, and your task is to find the largest 
or smallest M integers (and associated strings) 
in the input stream. You might imagine the 
stream to be financial transactions, where 
your interest is to find the big ones, or pesti¬ 
cide levels in an agricultural product, where 
your interest is to find the small ones, or re¬ 
quests for service, or results from a scientific 
experiment, or whatever. In some applications, the size of the input stream is so huge 
that it is best to consider it to be unbounded. One way to address this problem would 
be to sort the input stream and take the M largest keys from the result, but we have 
just stipulated that the input stream is too large for that. Another approach would be 
to compare each new key against the M largest seen so far, but that is also likely to be 
prohibitively expensive unless M is small. With priority queues, we can solve the prob¬ 
lem with the Mi nPQ client TopM on the next page provided that we can develop efficient 
implementations of both i nse rt 0 and del Mi n (). That is precisely our aim in this sec¬ 
tion. For the huge values of N that are likely to be encountered in our modern compu¬ 
tational infrastructure, these implementations can make the difference between being 
able to address such a problem and not having the resources to do it at all. 

Elementary implementations The basic data structures that we discussed in 
chapter l provide us with four immediate starting points for implementing priority 
queues. We can use an array or a linked list, kept in order or unordered. These imple¬ 
mentations are useful for small priority queues, situations where one of the two opera¬ 
tions are predominant, or situations where some assumptions can be made about the 
order of the keys involved in the operations. Since these implementations are elemen¬ 
tary, we will be content with brief descriptions here in the text and leave the code for 
exercises (see exercise 2.4.3). 

Array representation (unordered). Perhaps the simplest priority-queue implementa¬ 
tion is based on our code for pushdown stacks in section 2.1. The code for insert in the 
priority queue is the same as for push in the stack. To implement remove the maximum, 
we can add code like the inner loop of selection sort to exchange the maximum item 
with the item at the end and then delete that one, as we did with pop() for stacks. As 
with stacks, we can add resizing-array code to ensure that the data structure is always at 
least one-quarter full and never overflows. 


client 

order of growth 


time 

space 

sort client 

NlogN 

N 

PQ client using 

NM 

M 

elementary implementation 

PQ client using 
heap-based implementation 

N log M 

M 

Costs of finding the largest M in a stream of N items 
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A priority-queue client 


public class TopM 

{ 

public static void main(String[] args) 

{ // Print the top M lines in the input stream, 

int M = Integer.parselnt(args[0]); 

MinPQ<Transaction> pq = new MinPQ<Transaction>(M+l); 
while (Stdln.hasNextLine()) 

{ // Create an entry from the next line and put on the PQ. 

pq.insert(new Transaction(StdIn. readLineO)) ; 
if (pq.sizeO > M) 

pq.delMinO; // Remove minimum if M+l entries on the PQ. 
} // Top M entries are on the PQ. 

Stacl«Transaction> stack = new Stacl«Transaction>0 ; 
while C! pq - i sEmptyO) stack. push(pq. delMinO) ; 
for (Transaction t : stack) StdOut.println(t); 

} 

} 


Given an integer M from the command line and an input stream where each line contains a trans¬ 
action, this Mi nPQ client prints the M lines whose numbers are the highest. It does so by using our 


Transaction class (seepage 79, exercise 1 . 2 . 19 , and exercise 2 . 1 . 21 ) to build a priority queue using 

the numbers as keys, deleting the minimum after each 


% more tinyBatch.txt 


Turi ng 
vonNeumann 
Di jkstra 
vonNeumann 
Di jkstra 
Hoare 

vonNeumann 
Hoare 
Turi ng 
Thompson 
Turi ng 
Hoare 

vonNeumann 
Di jkstra 
Turi ng 
Hoare 


6/17/1990 

3/26/2002 

8/22/2007 

1/11/1999 

11/18/1995 

5/10/1993 

2/12/1994 

8/18/1992 

1/11/2002 

2/27/2000 

2/11/1991 

8/12/2003 

10/13/1993 

9/10/2000 

10/12/1993 

2/10/2005 


644.08 

4121.85 
2678.40 
4409.74 

837.42 

3229.27 

4732.35 
4381.21 

66.10 

4747.08 

2156.86 
1025.70 
2520.97 

708.95 

3532.36 
4050.20 


insertion once the size of the priority queue reaches M. 
Once all the transactions have been processed, the top M 
come off the priority queue in increasing order, so this 
code puts them on a stack, then iterates through the 
stack to reverse the order and print them in increasing 
order. 


% java TopM 5 < tinyBatch.txt 
Thompson 2/27/2000 4747.08 

vonNeumann 2/12/1994 4732.35 

vonNeumann 1/11/1999 4409.74 

Hoare 8/18/1992 4381.21 

vonNeumann 3/26/2002 4121.85 
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Array representation (ordered). Another approach is to add code for insert to move 
larger entries one position to the right, thus keeping the keys in the array in order (as in 
insertion sort). Thus, the largest element is always at the end, and the code for remove 
the maximum in the priority queue is the same as for pop in the stack. 

Linked-list representations. Similarly, we can start with our linked-list code for push¬ 
down stacks, modifying either the code for pop() to find and return the maximum or 
the code for push () to keep keys in reverse order and the code for popO to unlink and 
return the first (maximum) item on the list. 

Using unordered sequences is the prototypical lazy 
approach to this problem, where we defer doing 
work until necessary (to find the maximum); us¬ 
ing ordered sequences is the prototypical eager ap¬ 
proach to the problem, where we do as much work 
as we can up front (keep the list sorted on insertion) 
to make later operations efficient. 

The significant difference between implementing 
stacks or queues and implementing priority queues 
has to do with performance. For stacks and queues, 
we were able to develop implementations of all the 
operations that take constant time; for priority queues, all of the elementary imple¬ 
mentations just discussed have the property that either the insert or the remove the 
maximum operation takes linear time in the worst case. The heap data structure that we 
consider next enables implementations where both operations are guaranteed to be fast. 


data structure 


insert 


remove 

maximum 


ordered array 

N 

1 

unordered array 

1 

N 

heap 

log N 

log N 

impossible 

1 

1 


Order of growth of worst-case running time 
for priority-queue implementations 


operation argument return value size contents (unordered) 


contents (ordered) 


insert p 

insert Q 

insert E 

remove max 

insert X 

insert A 

insert |V| 

remove max 
insert p 

insert \_ 

insert E 

remove max 


1 

P 






P 





2 

P 

Q 





P 

Q 




3 

P 

Q 

E 




E 

P 

Q 



Q 2 

P 

E 





E 

P 




3 

P 

E 
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E 

P 

X 



4 

P 

E 

X 

A 



A 

E 

P 
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5 

P 

E 

X 

A 

M 


A 

E 

M 

P 

X 

X 4 

P 

E 

M 

A 



A 

E 

M 

P 


5 

P 

E 

M 

A 

P 


A 

E 

M 

P 

P 

6 

P 

E 

M 

A 

P 

L 

A 

E 

L 

M 

P 

7 

P 

E 

M 

A 

P 

L E 

A 

E 

E 

L 

M 

P 6 

E 

E 

M 

A 

P 

L 

A 

E 

E 

L 

M 

A sequence of operations 

on a priority queue 
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Heap definitions The binary heap is a data structure that can efficiently support 
the basic priority-queue operations. In a binary heap, the keys are stored in an array 
such that each key is guaranteed to be larger than (or equal to) the keys at two other 
specific positions. In turn, each of those keys must be larger than (or equal to) two ad¬ 
ditional keys, and so forth. This ordering is easy to see if we view the keys as being in 
a binary tree structure with edges from each key to the two keys known to be smaller. 

Definition. A binary tree is heap-ordered, if the key in each node is larger than or 
equal to the keys in that node’s two children (if any). 


Equivalently, the key in each node of a heap-ordered binary tree is smaller than or 
equal to the key in that node’s parent (if any). Moving up from any node, we get a 
nondecreasing sequence of keys; moving down from any node, we get a nonincreasing 
sequence of keys. In particular: 

Proposition O. The largest key in a heap-ordered binary tree is found at the root. 

Proof: By induction on the size of the tree. 


Binary heap representation. If we use a linked representation for heap-ordered binary 
trees, we would need to have three links associated with each key to allow travel up and 
down the tree (each node would have one pointer to its parent and one to each child). 
It is particularly convenient, instead, to use a complete binary tree like the one drawn at 
right. We draw such a structure by placing the root node 
and then proceeding down the page and from left to right, 
drawing and connecting two nodes beneath each node on 
the previous level until we have drawn N nodes. Complete 
trees provide the opportunity to use a compact array rep¬ 
resentation that does not involve explicit links. Specifically, 
we represent complete binary trees sequentially within an 
array by putting the nodes in level order, with the root at 
position 1, its children at positions 2 and 3, their children 
in positions 4, 5, 6, and 7, and so on. 
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Definition. A binary heap is a collection of keys arranged in a complete heap-or¬ 
dered binary tree, represented in level order in an array (not using the first entry). 


(For brevity, from now on we drop the “binary” 
modifier and use the term heap when referring 
to a binary heap.) In a heap, the parent of the 
node in position k is in position [k/2\ and, con¬ 
versely, the two children of the node in position 
k are in positions 2k and 2k + 1. Instead of using 
explicit links (as in the binary tree structures that 
we will consider in chapter 3 ), we can travel up 
and down by doing simple arithmetic on array 
indices: to move up the tree from a[k] we set k 
to k/ 2 ; to move down the tree we set k to 2 *k or 
2 *k+l. 

Complete binary trees represented as arrays 
(heaps) are rigid structures, but they have just 
enough flexibility to allow us to implement effi¬ 
cient priority-queue operations. Specifically, we 
will use them to develop logarithmic-time insert and remove the maximum implemen¬ 
tations. These algorithms take advantage of the capability to move up and down paths 
in the tree without pointers and have guaranteed logarithmic performance because of 
the following property of complete binary trees: 

Proposition P. The height of a complete binary tree of size N is |_ lg N J . 

Proof: The stated result is easy to prove by induction or by noting that the height 
increases by 1 when N is a power of 2. 


i 0123456789 10 11 
a[i] - TSRPNOAEIHG 
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Algorithms on heaps We represent a heap of size N in private array pq[] of 
length N + 1, with pq [0] unused and the heap in pq [1] through pq [N] . As for sorting 
algorithms, we access keys only through private helper functions 1 ess () and exch (), 
but since all items are in the instance 
variable pq [], we use the more com¬ 
pact implementations on the next 
page that do not involve passing the 
array name as a parameter. The heap 
operations that we consider work by 
first making a simple modification 
that could violate the heap condition, 
then traveling through the heap, modifying the heap as required to ensure that the heap 
condition is satisfied everywhere. We refer to this process as reheapifying, or restoring 
heap order. 

There are two cases. When the priority of some node is increased (or a new node is 
added at the bottom of a heap), we have to travel up the heap to restore the heap order. 
When the priority of some node is decreased (for example, if we replace the node at 
the root with a new node that has a smaller key), we have to travel down the heap to 
restore the heap order. First, we will consider how to implement these two basic auxil¬ 
iary operations; then, we shall see how to use them to implement insert and remove the 
maximum. 

Bottom-up reheapify (swim). If the heap order is violated because a node’s key be¬ 
comes larger than that node’s parent’s key, then we can make progress toward fixing 

the violation by exchanging the node with 
its parent. After the exchange, the node is 
larger than both its children (one is the old 
parent, and the other is smaller than the old 
parent because it was a child of that node) 
but the node may still be larger than its par¬ 
ent. We can fix that violation in the same 
way, and so forth, moving up the heap until 
we reach a node with a larger key, or the 
root. Coding this process is straightforward 
when you keep in mind that the parent of 
the node at position k in a heap is at po¬ 
sition k/2. The loop in swim() preserves 
the invariant that the only place the heap 




private boolean less(int i, int j) 

{ return pq[i] . compareTo(pq[j]) <0; } 

private void exch (int i, int j) 

{ Key t = pq[i] ; pq[i] = pq[j] ; pq[j] = t; } 

Compare and exchange methods for heap implementations 
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order could be violated is when the node at 
position k might be larger than its parent. 

Therefore, when we get to a place where that 
node is not larger than its parent, we know 
that the heap order is satisfied throughout 

the heap. To jus¬ 
tify the method’s 
name, we think 
of the new node, 
having too large a 

key, as having to swim to a higher level in the heap. 

Top-down reheapify (sink). If the heap order is violated be¬ 
cause a node’s key becomes smaller than one or both of that 
node’s children’s keys, then we can make progress toward fix¬ 
ing the violation by exchanging the node with the larger of its 
two children. This switch may cause a violation at the child; 
we fix that violation in the same way, and so forth, moving 
down the heap until we reach a node with both children 
smaller (or equal), or the bottom. The code again follows di¬ 
rectly from 
the fact that 

the children of the node at position 
k in a heap are at positions 2 k and 
2 k+l. To justify the method’s name, 
we think about the node, having too 
small a key, as having to sink to a low¬ 
er level in the heap. 

if we imagine the heap to represent 
a cutthroat corporate hierarchy, with 
each of the children of a node repre¬ 
senting subordinates (and the parent 
representing the immediate superior), then these operations have amusing interpreta¬ 
tions. The swi m() operation corresponds to a promising new manager arriving on the 
scene, being promoted up the chain of command (by exchanging jobs with any lower- 
qualified boss) until the new person encounters a higher-qualified boss. The sinkO 
operation is analogous to the situation when the president of the company resigns and 
is replaced by someone from the outside. If the president’s most powerful subordinate 


private void sink(int k) 

{ 

while (2*k <= N) 

{ 

int j = 2*k; 

if (j < N && less(j, j+1)) j++; 
if (!less(k, j)) break; 
exch(k, j); 

k = j; 

} 


Top-down reheapify (sink) implementation 


violates heap order 



Top-down reheapify (sink) 


private void swim(int k) 

{ 

while (k > 1 && less(k/2, k)) 
{ 

exch(k/2, k); 
k = l</2; 

} 

} 


Bottom-up reheapify (swim) implementation 
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is stronger than the new person, they exchange jobs, and we move down the chain of 
command, demoting the new person and promoting others until the level of compe¬ 
tence of the new person is reached, where there is no higher-qualified subordinate. 
These idealized scenarios may rarely be seen in the real world, but they may help you 
better understand basic operation on heaps. 

These si nk() and swim() operations provide the basis for efficient implementation 
of the priority-queue API, as diagrammed below and implemented in algorithm 2.6 
on the next page. 

Insert. We add the new key at the end of the array, increment the size of the heap, 
and then swim up through the heap with that key to restore the heap condition. 

Remove the maximum. We take the largest key off the top, put the item from 
the end of the heap at the top, decrement the size of the heap, and then sink down 
through the heap with that key to restore the heap condition. 

algorithm 2.6 solves the basic problem that we posed at the beginning of this section: 
it is a priority-queue API implementation for which both insert and delete the maxi¬ 
mum are guaranteed to take time logarithmic in the size of the queue. 





Heap operations 
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ALGORITHM 2.6 Heap priority queue 


public class MaxPQ<Key extends Comparable<Key» 

{ 

private Key[] pq; // heap-ordered complete binary tree 

private int N = 0; // in pq[l. .N] with pq[0] unused 

public MaxPQ(int maxN) 

{ pq = (Key[]) new Comparable[maxN+1]; } 

public boolean isEmptyO 
{ return N == 0; } 

public int size() 

{ return N; } 


public void insert(Key v) 

{ 

pq[++N] = v; 
swim(N); 

} 

public Key delMaxO 

{ 

Key max = pq[1]; 
exch(l, N--); 
pq[N+l] = nul1; 
sink(l); 
return max; 

} 


// Retrieve max key from top. 
// Exchange with last item. 

// Avoid loitering. 

// Restore heap property. 


// See pages 145-147 for implementations of these helper methods. 

private boolean less(int i, int j) 

private void exch(int i, int j) 

private void swim(int k) 

private void sink(int k) 


The priority queue is maintained in a heap-ordered complete binary tree in the array pq [] with 
pq [0] unused and the N keys in the priority queue in pq [1] through pq [N] . To implement i nsert(), 
we increment N, add the new element at the end, then use swim() to restore the heap order. For 
delMaxO, we take the value to be returned from pq[1] , then move pq[N] to pq[1] , decrement the 
size of the heap, and use si nk() to restore the heap condition. We also set the now-unused position 
pq[N+l] to null to allow the system to reclaim the memory associated with it. Code for dynamic 
array resizing is omitted, as usual (see section 1.3). See exercise 2.4.19 for the other constructors. 
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Proposition Q. In an JV-key priority queue, the heap al¬ 
gorithms require no more than 1 + lg iV compares for in¬ 
sert and no more than 2 lg N compares for remove the 
maximum. 

Proof: By proposition p, both operations involve mov¬ 
ing along a path between the root and the bottom of the 
heap whose number of links is no more than lg N. The 
remove the maximum operation requires two compares 
for each node on the path (except at the bottom): one 
to find the child with the larger key, the other to decide 
whether that child needs to be promoted. 


For typical applications that require a large number of in¬ 
termixed insert and remove the maximum operations in a 
large priority queue, proposition q represents an important 
performance breakthrough, summarized in the table shown 
on page 312. Where elementary implementations using an or¬ 
dered array or an unordered array require linear time for one 
of the operations, a heap-based implementation provides a 
guarantee that both operations complete in logarithmic time. 
This improvement can make the difference between solving a 
problem and not being able to address it at all. 

Multiway heaps. It is not difficult to modify our code to 
build heaps based on an array representation of complete 
heap-ordered ternary trees, with an entry at position k larger 
than or equal to entries at positions 3 k~ 1, 3k, and 3 k +1 and 
smaller than or equal to entries at position |_(k+l)/3], for all 
indices between 1 and N in an array of N items, and not much 
more difficult to use d-ary heaps for any given d. There is a 
tradeoff between the lower cost from the reduced tree height 
(log rf N) and the higher cost of finding the largest of the d 
children at each node. This tradeoff is dependent on details 
of the implementation and the expected relative frequency of 
operations. 


insert P 



insert Q 


insert E 


remove max (Q) 


insert X 


insert A 




insert P 


insert L 


insert E 


remove max (P) 



Priority queue operations in a heap 
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Array resizing. We can add a no-argument constructor, code for array doubling in 
i nsert(), and code for array halving in del Max O, just as we did for stacks in section 
1 . 3 . Thus, clients need not be concerned about arbitrary size restrictions. The logarith¬ 
mic time bounds implied by proposition q are amortized when the size of the priority 
queue is arbitrary and the arrays are resized (see exercise 2 . 4 . 22 ). 

Immutability of keys. The priority queue contains objects that are created by clients 
but assumes that client code does not change the keys (which might invalidate the 
heap-order invariant). It is possible to develop mechanisms to enforce this assumption, 
but programmers typically do not do so because they complicate the code and are likely 
to degrade performance. 

Index priority queue. In many applications, it makes sense to allow clients to refer 
to items that are already on the priority queue. One easy way to do so is to associate 
a unique integer index with each item. Moreover, it is often the case that clients have 
a universe of items of a known size N and perhaps are using (parallel) arrays to store 
information about the items, so other unrelated client code might already be using an 
integer index to refer to items. These considerations lead us to the following API: 

public class IndexMi nPQ<Item extends Comparable<Item» 



IndexMinPQ(int maxN) 

create a priority queue of capacity maxN 
with possible indices between 0 and maxN-1 

voi d 

insert(int k, Item item) 

insert i tern; associate it with k 

voi d 

change(int k, Item item) 

change the item associated with k to i tem 

boolean 

contains(int k) 

is k associated with some item? 

voi d 

delete(int k) 

remove k and its associated item 

Item 

mi n() 

return a minimal item 

int 

mi nlndexO 

return a minimal item’s index 

int 

delMinO 

remove a minimal item and return its index 

boolean 

i sEmptyO 

is the priority queue empty? 

int 

size() 

number of items in the priority queue 


API for a generic priority queue with associated indices 
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A useful way of thinking of this data type is as implementing an array, but with fast ac¬ 
cess to the smallest entry in the array. Actually it does even better—it gives fast access 
to the minimum entry in a specified subset of an array’s entries (the ones that have been 
inserted. In other words, you can think of an IndexMinPQ named pq as representing 
a subset of an array pq [0.. N-l] of items. Think of the call pq. i nsertfk, item) as 
adding k to the subset and setting pq[k] = item and the call pq.change(k, item) 
as setting pq [k] = i tern, both also maintaining data structures needed to support the 
other operations, most importantly delMinO (remove and return the index of the 
minimum key) and change () (change the item associated with an index that is already 
in the data structure—just as in pq [i ] = i tern). These operations are 
important in many applications and are enabled by our ability to refer 
to the key (with the index), exercise 2 . 4.33 describes how to extend 
algorithm 2.6 to implement index priority queues with remarkable 
efficiency and with remarkably little code. Intuitively, when an item in 
the heap changes, we can restore the heap invariant with a sink opera¬ 
tion (if the key increases) and a swim operation (if the key decreases). 

To perform the operations, we use the index to find the item in the 
heap. The ability to locate an item in the heap also allows us to add the 
del ete () operation to the API. 


This discussion is for a minimum-oriented queue; as usual, we also 
implement on the booksite a maximum-oriented version IndexMaxPQ. 

Index priority-queue client. The IndexMi nPQ client Multi way on page 322 solves the 
multiway merge problem: it merges together several sorted input streams into one 
sorted output stream. This problem arises in many applications: the streams might 
be the output of scientific instruments (sorted by time), lists of information from the 
web such as music or movies (sorted by title or artist name), commercial transactions 
(sorted by account number or time), or whatever. If you have the space, you might just 
read them all into an array and sort them, but with a priority queue, you can read input 
streams and put them in sorted order on the output no matter how long they are. 


Proposition Q (continued). In an index priority queue of size N, 
the number of compares required is proportional to at most log N 
for insert, change priority, delete, and remove the minimum. 

Proof: Immediate from inspection of the code and the fact that all 
paths in a heap are of length at most ~lg N. 


operation 

order of 
growth of 
number of 
compares 

i nsert() 

log N 

change() 

log N 

contains() 

1 

del ete() 

log N 

min() 

1 

mi nlndex() 

1 

delMi n() 

log N 


Worst-case costs for 
an /V-item heap-based 
indexed priority queue 
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Multiway merge priority-queue client 


public class Multiway 

{ 

public static void merge(In[] streams) 

{ 

int N = streams.1ength; 

IndexMinPQ<String> pq = new IndexMinPQ<String>(N); 

for (int i = 0; i < N; i++) 
if (!streams[i].isEmptyO) 

pq.insert(i, streams[i].readString()); 

while (!pq.isEmptyO) 

{ 

StdOut.println(pq.minO); 
int i = pq.delMinO ; 
if (!streams[i].isEmptyO) 

pq.insert(i, streams[i].readString()); 

} 

} 

public static void main(String[] args) 

{ 

int N = args.length; 

In[] streams = new In[N]; 
for (int i = 0; i < N; i++) 

streams[i] = new In(args[i]); 
merge(streams); 

} 

} 


This IndexMi nPQ client merges together the sorted input stream given as command-line arguments 
into a single sorted output stream on standard output (see text). Each stream index is associated with 
a key (the next string in the stream). After initialization, it enters a loop that prints the smallest string 
in the queue and removes the corresponding entry, then adds a new entry for the next string in that 
stream. For economy, the output is shown on one line below—the actual output is one string per line. 

% more ml.txt 
ABCFGIIZ 
% more m2.txt 
B D H P Q Q 
% more m3.txt 
A B E F 3 N 


% java Multiway ml.txt m2.txt m3.txt 
AABBBCDEFFGHII1NPQQZ 
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Heapsort We can use any priority queue to develop a sorting method. We insert 
all the items to be sorted into a minimum-oriented priority queue, then repeatedly use 
remove the minimum to remove them all in order. Using a priority queue represented as 
an unordered array in this way corresponds to doing a selection sort; using an ordered 
array corresponds to doing an insertion sort. What sorting method do we get if we use 
a heap? An entirely different one! Next, we use the heap to develop a classic elegant sort¬ 
ing algorithm known as heapsort. 

Heapsort breaks into two phases: heap construction, where we reorganize the original 
array into a heap, and the sortdown, where we pull the items out of the heap in decreas¬ 
ing order to build the sorted result. For consistency with the code we have studied, we 
use a maximum-oriented priority queue and repeatedly remove the maximum. Focus¬ 
ing on the task of sorting, we abandon the notion of hiding the representation of the 
priority queue and use swi m() and si nk() directly. Doing so allows us to sort an array 
without needing any extra space, by maintaining the heap within the array to be sorted. 

Heap construction. How difficult is the process of building a heap from N given items? 
Certainly we can accomplish this task in time proportional to N log N, by proceeding 
from left to right through the array, using swi m() to ensure that the items to the left of 
the scanning pointer make up a heap-ordered complete tree, like successive priority- 
queue insertions. A clever method that is much more efficient is to proceed from right 
to left, using si nk() to make subheaps as we go. Every position in the array is the root 
of a small subheap; si nk() works for such subheaps, as well. If the two children of a 
node are heaps, then calling si nk() on that node makes the subtree rooted at the par¬ 
ent a heap. This process establishes the heap order inductively. The scan starts halfway 
back through the array because we can skip the subheaps of size 1. The scan ends at 
position 1, when we finish building the heap with one call to si nk() . As the first phase 
of a sort, heap construction is a bit counterintuitive, because its goal is to produce a 
heap-ordered result, which has the largest item first in the array (and other larger items 
near the beginning), not at the end, where it is destined to finish. 

Proposition R. Sink-based heap construction uses fewer than 2 N compares and 
fewer than N exchanges to construct a heap from N items. 

Proof: This fact follows from the observation that most of the heaps processed are 
small. For example, to build a heap of 127 elements, we process 32 heaps of size 3, 

16 heaps of size 7, 8 heaps of size 15, 4 heaps of size 31, 2 heaps of size 63, and 1 
heap of size 127, so 32T + 16-2 + 8-3 + 4-4 + 2-5 + T 6 = 120 exchanges (twice as 
many compares) are required (at worst). See exercise 2 . 4.20 for a complete proof. 
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ALGORITHM 2.7 Heapsort 


public static void sort(Comparable[] a) 

{ 

int N = a.length; 
for (int k = N/2; k >= 1; k--) 
sink(a, k, N); 
while (N > 1) 

{ 

exch(a, 1, N--); 
sink(a, 1, N); 

} 

} 


This code sorts a[l] through a[N] using the sink() method (modified to take a[] and N as argu¬ 
ments). The for loop constructs the heap; then the while loop exchanges the largest element a[l] 
with a [N] and then repairs the heap, continuing until the heap is empty. Decrementing the array in¬ 
dices in the implementations of exch () and 1 ess () gives an implementation that sorts a [0] through 
a [N-l] , consistent with our other sorts. 
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heap construction 


sortdown 



starting point (arbitrary order) 


starting point (heap-ordered) 
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exch(l, 2) 
sink(l, 1) 



Heapsort: constructing (left) and sorting down (right) a heap 
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Visual trace of heapsort 


Sortdown. Most of the work during heapsort is done during 
the second phase, where we remove the largest remaining item 
from the heap and put it into the array position vacated as the 
heap shrinks. This process is a bit like selection sort (taking the 
items in decreasing order instead of in increasing order), but it 
uses many fewer compares because the heap provides a much 
more efficient way to find the largest item in the unsorted part 
of the array. 

Proposition S. Heapsort uses fewer than IN lg iV + 2 N 
compares (and half that many exchanges) to sort N items. 

Proof: The 2 N term covers the cost of heap construc¬ 
tion (see proposition r). The 2Nlg N term follows from 
bounding the cost of each sink operation during the sort- 
down by 2lg N (see proposition pq). 


algorithm 2.7 is a full implementation based on these ideas, 
the classical heapsort algorithm, which was invented by J. W. 
J. Williams and refined by R. W. Floyd in 1964. Although the 
loops in this program seem to do different tasks (the first 
constructs the heap, and the second destroys the heap for the 
sortdown), they are both built around the si nk() method. We 
provide an implementation outside of our priority-queue API 
to highlight the simplicity of the sorting algorithm (eight lines 
of code for sort() and another eight lines of code for si nk()) 
and to make it an in-place sort. 

As usual, you can gain some insight into the operation of the 
algorithm by studying a visual trace. At first, the process seems 
to do anything but sort, because large items are moving to the 
beginning of the array as the heap is being constructed. But 
then the method looks more like a mirror image of selection 
sort (except that it uses far fewer compares). 

As for all of the other methods that we have studied, various 
people have investigated ways to improve heap-based priority- 
queue implementations and heapsort. We now briefly consider 
one of them. 
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Sink to the bottom, then swim. Most items reinserted into the heap during sortdown 
go all the way to the bottom. Floyd observed in 1964 that we can thus save time by 
avoiding the check for whether the item has reached its position, simply promoting 
the larger of the two children until the bottom is reached, then moving back up the 
heap to the proper position. This idea cuts the number of compares by a factor of 2 as¬ 
ymptotically—close to the number used by mergesort (for a randomly-ordered array). 
The method requires extra bookkeeping, and it is useful in practice only when the cost 
of compares is relatively high (for example, when we are sorting items with strings or 
other types of long keys). 

heapsort is significant in the study of the complexity of sorting (see page 279 ) because 
it is the only method that we have seen that is optimal (within a constant factor) in its 
use of both time and space—it is guaranteed to use ~2N lg N compares and constant 
extra space in the worst case. When space is very tight (for example, in an embedded 
system or on a low-cost mobile device) it is popular because it can be implemented 
with just a few dozen lines (even in machine code) while still providing optimal per¬ 
formance. However, it is rarely used in typical applications on modern systems because 
it has poor cache performance: array entries are rarely compared with nearby array 
entries, so the number of cache misses is far higher than for quicksort, mergesort, and 
even shellsort, where most compares are with nearby entries. 

On the other hand, the use of heaps to implement priority queues plays an increas¬ 
ingly important role in modern applications, because it provides an easy way to guar¬ 
antee logarithmic running time for dynamic situations where large numbers of insert 
and remove the maximum operations are intermixed. We will encounter several ex¬ 
amples later in this book. 
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| Q&A 

Q. I’m still not clear on the purpose of priority queues. Why exactly don’t we just sort 
and then consider the items in increasing order in the sorted array? 

A. In some data-processing examples such as TopM and Mul ti way, the total amount of 
data is far too large to consider sorting (or even storing in memory). If you are looking 
for the top ten entries among a billion items, do you really want to sort a billion-entry 
array? With a priority queue, you can do it with a ten-entry priority queue. In other ex¬ 
amples, all the data does not even exist together at any point in time: we take something 
from the priority queue, process it, and as a result of processing it perhaps add some 
more things to the priority queue. 

Q. Why not use Comparabl e, as we do for sorts, instead of the generic Item in MaxPQ? 

A. Doing so would require the client to cast the return value of del Max() to an actual 
type, such as Stri ng. Generally, casts in client code are to be avoided. 

Q. Why not use a[0] in the heap representation? 

A. Doing so simplifies the arithmetic a bit. It is not difficult to implement the heap 
methods based on a 0-based heap where the children of a[0] are a[l] and a [2], the 
children of a[l] are a[3] and a[4], the children of a [2] are a[5] and a[6], and 
so forth, but most programmers prefer the simpler arithmetic that we use. Also, us¬ 
ing a [0] as a sentinel value (in the parent of a [1] ) is useful in some heap applications. 

Q. Building a heap in heapsort by inserting items one by one seems simpler to me than 
the tricky bottom-up method described on page 323 in the text. Why bother? 

A. For a sort implementation, it is 20 percent faster and requires half as much tricky 
code (no swimO needed). The difficulty of understanding an algorithm has not neces¬ 
sarily much to do with its simplicity, or its efficiency. 

Q. What happens if I leave off the extends Comparabl e<Key> phrase in an implemen¬ 
tation like MaxPQ ? 

A. As usual, the easiest way for you to answer a question of this sort for yourself is to 
simply try it. If you do so for MaxPQ you will get a compile-time error: 

MaxPQ.java:21: cannot find symbol 
symbol : method compareTo(Item) 

which is java’s way of telling you that it does not know about compareToO in Item 
because you neglected to declare that Item extends Comparabl e<Item>. 
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EXERCISES 


2.4.1 Suppose that the sequence PRIO*R**I*T*Y***QUE*** 
U * E (where a letter means insert and an asterisk means remove the maximum) is ap¬ 
plied to an initially empty priority queue. Give the sequence of letters returned by the 
remove the maximum operations. 

2.4.2 Criticize the following idea: To implement/md the maximum in constant time, 
why not use a stack or a queue, but keep track of the maximum value inserted so far, 
then return that value for find the maximum ? 

2.4.3 Provide priority-queue implementations that support insert and remove the 
maximum, one for each of the following underlying data structures: unordered ar¬ 
ray, ordered array, unordered linked list, and linked list. Give a table of the worst-case 
bounds for each operation for each of your four implementations. 

2.4.4 Is an array that is sorted in decreasing order a max-oriented heap? 

2.4.5 Give the heap that results when the keys EASYQUESTION are inserted 
in that order into an initially empty max-oriented heap. 

2.4.6 Using the conventions of exercise 2 . 4 . 1 , give the sequence of heaps produced 
when the operations P R I 0 * R * * I * T * Y * * * Q U E * * * U * E are 
performed on an initially empty max-oriented heap. 

2.4.7 The largest item in a heap must appear in position 1, and the second largest must 
be in position 2 or position B. Give the list of positions in a heap of size 31 where the 
kth largest (z) can appear, and (**) cannot appear, for k=2 , 3 , 4 (assuming the values to 
be distinct). 

2.4.8 Answer the previous exercise for the kth smallest item. 

2.4.9 Draw all of the different heaps that can be made from the five keys A B C D E, 
then draw all of the different heaps that can be made from the five keys A A A B B. 

2.4.10 Suppose that we wish to avoid wasting one position in a heap-ordered array 
pq □, putting the largest value in pq [0], its children in pq [1] and pq [2], and so forth, 
proceeding in level order. Where are the parents and children of pq [k] ? 

2.4.11 Suppose that your application will have a huge number of insert operations, but 
only a few remove the maximum operations. Which priority-queue implementation do 
you think would be most effective: heap, unordered array, or ordered array? 
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EXERCISES (continued) 


2.4.12 Suppose that your application will have a huge number of find, the maximum 
operations, but a relatively small number of insert and remove the maximum operations. 
Which priority-queue implementation do you think would be most effective: heap, 
unordered array, or ordered array? 

2.4.13 Describe a way to avoid the j < N test in si nk(). 

2.4.14 What is the minimum number of items that must be exchanged during a re¬ 
move the maximum operation in a heap of size N with no duplicate keys? Give a heap 
of size 15 for which the minimum is achieved. Answer the same questions for two and 
three successive remove the maximum operations. 

2.4.15 Design a linear-time certification algorithm to check whether an array pq [] is 
a min-oriented heap. 

2.4.16 For N= 32, give arrays of items that make heapsort use as many and ns few com¬ 
pares as possible. 

2.4.17 Prove that building a minimum-oriented priority queue of size k then doing N 
— k replace the minimum (insert followed by remove the minimum) operations leaves the 
k largest of the N items in the priority queue. 

2.4.18 In MaxPQ, suppose that a client calls i nsert() with an item that is larger than 
all items in the queue, and then immediately calls delMax(). Assume that there are 
no duplicate keys. Is the resulting heap identical to the heap as it was before these op¬ 
erations? Answer the same question for two i nsert() operations (the first with a key 
larger than all keys in the queue and the second for a key larger than that one) followed 
by two delMaxO operations. 

2.4.19 Implement the constructor for MaxPQ that takes an array of items as argument, 
using the bottom-up heap construction method described on page 323 in the text. 

2.4.20 Prove that sink-based heap construction uses fewer than 2 N compares and 
fewer than N exchanges. 
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CREATIVE PROBLEMS 


2.4.21 Elementary data structures. Explain how to use a priority queue to implement 
the stack, queue, and randomized queue data types from chapter i. 

2.4.22 Array resizing. Add array resizing to MaxPQ, and prove bounds like those of 
proposition q for array accesses, in an amortized sense. 

2.4.23 Multiway heaps. Considering the cost of compares only, and assuming that 
it takes t compares to find the largest of t items, find the value of t that minimizes the 
coefficient of Nig N in the compare count when a t- ary heap is used in heapsort. First, 
assume a straightforward generalization of si nk(); then, assume that Floyd’s method 
can save one compare in the inner loop. 

2.4.24 Priority queue with explicit links. Implement a priority queue using a heap- 
ordered binary tree, but use a triply linked structure instead of an array. You will need 
three links per node: two to traverse down the tree and one to traverse up the tree. Your 
implementation should guarantee logarithmic running time per operation, even if no 
maximum priority-queue size is known ahead of time. 

2.4.25 Computational number theory. Write a program CubeSum .java that prints out 
all integers of the form a 3 + b 3 where a and b are integers between 0 and N in sorted 
order, without using excessive space. That is, instead of computing an array of the N 2 
sums and sorting them, build a minimum-oriented priority queue, initially containing 
(0 3 ,0,0), (l 3 ,1,0), (2 3 ,2,0), ... , (NTN, 0). Then, while the priority queue is nonempty, 
remove the smallest itemfz 3 +j 3 , i,j), print it, and then, if j < N, insert the item (i 3 + (j+1 ) 3 , 
i, j+1). Use this program to find all distinct mintegers a, b, c, and d between 0 and 10 6 
such that a 3 + b 3 = c 3 + d 3 . 

2.4.26 Heap without exchanges. Because the exch() primitive is used in the si nk() 
and swim() operations, the items are loaded and stored twice as often as necessary. 
Give more efficient implementations that avoid this inefficiency, a la insertion sort (see 
exercise 2 . 1 . 25 ). 

2.4.27 Find the minimum. Add a minO method to MaxPQ. Your implementation 
should use constant time and constant extra space. 

2.4.28 Selection filter. Write a TopM client that reads points (x, y, z) from standard in¬ 
put, takes a value M from the command line, and prints the M points that are closest to 
the origin in Euclidean distance. Estimate the running time of your client for N = 10 8 
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CREATIVE PROBLEMS (continued) 


and M= 10 4 . 

2.4.29 Min/max priority queue. Design a data type that supports the following opera¬ 
tions: insert, delete the maximum, and delete the minimum (all in logarithmic time); and 
find the maximum and find the minimum (both in constant time). Hint-. Use two heaps. 

2.4.30 Dynamic median-finding. Design a data type that supports insert in logarith¬ 
mic time, find the median in constant time, and delete the median in logarithmic time. 
Hint-. Use a min-heap and a max-heap. 

2.4.31 Fast insert. Develop a compare-based implementation of the Mi nPQ API such 
that insert uses ~ log log N compares and delete the minimum uses ~2log N compares. 
Hint: Use binary search on parent pointers to find the ancestor in swi m(). 

2.4.32 Lower bound. Prove that it is impossible to develop a compare-based imple¬ 
mentation of the Mi nPQ API such that both insert and delete the minimum guarantee to 
use ~N log log N compares. 

2.4.33 Index priority-queue implementation. Implement the basic operations in the 
index priority-queue API on page 320 by modifying algorithm 2.6 as follows: Change 
pq [] to hold indices, add an array keys [] to hold the key values, and add an array qp [] 
that is the inverse ofpq[] — qp [i ] gives the position of i in pq [] (the index j such that 
pq [ j ] is i ). Then modify the code in algorithm 2.6 to maintain these data structures. 
Use the convention that qp [i ] = -1 if i is not on the queue, and include a method 
contai ns() that tests this condition. You need to modify the helper methods exch() 
and lessQ but not sinkQ or swimQ. 
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Partial solution: 


public class IndexMinPQcKey extends Comparable<Key» 

{ 


private int N; // 

private int[] pq; // 

private int[] qp; // 

private Key[] keys; // 


public IndexMinPQ(int maxN) 

{ 


number of elements on PQ 
binary heap using 1-based indexing 
inverse: qp [pq[i]] = pq [qp [i]] = i 
items with priorities 


keys = (Key[]) new Comparable[maxN + 
[maxN + 1]; 

[maxN + 1]; 

=0; i <= maxN; i++) qp[i] 


0 ; } 


1 ] 



pq 

= new 


qp 

= new 

} 

for 

(int i : 

public 

boolean 

{ 

return N == 


-l; 


public boolean contains(int k) 

{ return qp[k] != -1; } 

public void insert(int k, Key key) 

{ 

N++; 

qp[k] = N; 
pq [N] = k; 
keys[k] = key; 
swim(N); 

} 

public Item min() 

{ return keys[pq[l]]; } 

public int delMinO 

{ 

int indexOfMin = pq[1]; 
exch(l, N--); 
sink(l); 

keys[pq[N+l]] = null; 
qp [pq [N+l]] = -1; 
return indexOfMin; 
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CREATIVE PROBLEMS (continued) 


2.4.34 Index priority-queue implementation (additional operations). Addmi nlndexO, 
changeO, anddeleteO to your implementation of exercise 2 . 4 . 33 . 

Solution: 

public int minlndexO 
{ return pq[1]; } 

public void change(int k, Item item) 

{ 

keys[k] = key; 
swim(qp[k]); 
sink(qp[k]); 

} 

public void delete(int k) 

{ 

exch(k, N--); 
swim(qp[k]); 
sink(qp[k]); 
keys[pq[N+l]] = null; 
qp [pq[N+l]] = -1; 

} 

2.4.35 Sampling from a discrete probability distribution. Write a class Sample with a 
constructor that takes an array p [] of doubl e values as argument and supports the fol¬ 
lowing two operations: randomO —return an index i with probability p [i ] /T (where 
T is the sum of the numbers in p [])—and change(i , v) —change the value of p [ i ] 
to v. Hint: Use a complete binary tree where each node has implied weight p [i ]. Store 
in each node the cumulative weight of all the nodes in its subtree. To generate a ran¬ 
dom index, pick a random number between 0 and T and use the cumulative weights to 
determine which branch of the subtree to explore. When updating p [i ], change all of 
the weights of the nodes on the path from the root to i. Avoid explicit pointers, as we 
do for heaps. 
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EXPERIMENTS 


2.4.36 Performance driver I. Write a performance driver client program that uses in¬ 
sert to fill a priority queue, then uses remove the maximum to remove half the keys, then 
uses insert to fill it up again, then uses remove the maximum to remove all the keys, doing 
so multiple times on random sequences of keys of various lengths ranging from small to 
large; measures the time taken for each run; and prints out or plots the average running 
times. 

2.4.37 Performance driver II. Write a performance driver client program that uses in¬ 
sert to fill a priority queue, then does as many remove the maximum and insert opera¬ 
tions as it can do in 1 second, doing so multiple times on random sequences of keys of 
various lengths ranging from small to large; and prints out or plots the average number 
of remove the maximum operations it was able to do. 

2.4.38 Exercise driver. Write an exercise driver client program that uses the methods 
in our priority-queue interface of algorithm 2.6 on difficult or pathological cases that 
might turn up in practical applications. Simple examples include keys that are already 
in order, keys in reverse order, all keys the same, and sequences of keys having only two 
distinct values. 

2.4.39 Cost of construction. Determine empirically the percentage of time heapsort 
spends in the construction phase for N= 10 3 , 10 6 , and 10 9 . 

2.4.40 Floyd s method. Implement a version of heapsort based on Floyd’s sink-to-the- 
bottom-and-then-swim idea, as described in the text. Count the number of compares 
used by your program and the number of compares used by the standard implementa¬ 
tion, for randomly ordered distinct keys with N= 10 3 , 10 6 , and 10 9 . 

2.4.41 Multiway heaps. Implement a version of heapsort based on complete heap- 
ordered 3-ary and 4-ary trees, as described in the text. Count the number of compares 
used by each and the number of compares used by the standard implementation, for 
randomly ordered distinct keys with N= 10 3 , 10 6 , and 10 9 . 

2.4.42 Preorder heaps. Implement a version of heapsort based on the idea of repre¬ 
senting the heap-ordered tree in preorder rather than in level order. Count the number 
of compares used by your program and the number of compares used by the standard 
implementation, for randomly ordered keys with N= 10 3 , 10 6 , and 10 9 . 
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sorting algorithms and priority queues are widely used in a broad variety of ap¬ 
plications. Our purpose in this section is to briefly survey some of these applications, 
consider ways in which the efficient methods that we have considered play a critical role 
in such applications, and discuss some of the steps needed to make use of our sort and 
priority-queue code. 

A prime reason why sorting is so useful is that it is much easier to search for an item 
in a sorted array than in an unsorted one. For over a century, people found it easy to 
look up someone’s phone number in a phone book where items are sorted by last name. 
Now digital music players organize song files by artist name or song title; search engines 
display search results in descending order of importance; spreadsheets display columns 
sorted by a particular field; matrix-processing packages sort the real eigenvalues of a 
symmetric matrix in descending order; and so forth. Other tasks are also made easier 
once an array is in sorted order: from looking up an item in the sorted index in the back 
of this book; to removing duplicates from a long list such as a mailing list, a list of vot¬ 
ers, or a list of websites; to performing statistical calculations such as removing outliers, 
finding the median, or computing percentiles. 

Sorting also arises as a critical subproblem in many applications that appear to have 
nothing to do with sorting at all. Data compression, computer graphics, computational 
biology, supply-chain management, combinatorial optimization, social choice, and 
voting are but a few of many examples. The algorithms that we have considered in this 
chapter play a critical role in the development of effective algorithms in each of the later 
chapters in this book. 

Most important is the system sort, so we begin by considering a number of practical 
considerations that come into play when building a sort for use by a broad variety of 
clients. While some of these topics are specific to Java, they each reflect challenges that 
need to be met in any system. 

Our primary purpose is to demonstrate that, even though we have used mechanisms 
that are relatively simple, the sorting implementations that we are studying are widely 
applicable. The list of proven applications of fast sorting algorithms is vast, so we can 
consider just a small fraction of them: some scientific, some algorithmic, and some 
commercial. You will find many more examples in the exercises, and many more than 
that on the booksite. Moreover, we will often refer back to this chapter to effectively ad¬ 
dress the problems that we later consider in this bookl 
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Sorting various types of data Our implementations sort arrays of Comparabl e 
objects. This Java convention allows us to use Java’s callback mechanism to sort arrays 
of objects of any type that implements the Comparable interface. As described in 
section 2 . 1 , implementing Comparable amounts to defining a compareToO method 
that implements a natural ordering for the type. We can use our code immediately to 
sort arrays of type String, Integer, Double, and other types such as File and URL, 
because these data types all implement Comparabl e. Being able to use the same code 
for all of those types is convenient, but typical applications involve working with data 
types that are defined for use within the application. Accordingly it is common to im¬ 
plement a compareToO method for user-defined data types, so that they implement 
Comparable, thus enabling client code to sort arrays of that type (and build priority 
queues of values of that type). 

Transaction example. A prototypical breeding ground for sorting applications is 
commercial data processing. For example, imagine that a company engaged in internet 
commerce maintains a record for each transaction involving a customer account that 
contains all of the pertinent information, such as the customer name, date, amount, 
and so forth. Nowadays, a successful company needs to be able to handle millions and 
millions of such transactions. As we saw in exercise 2.1.21, it is reasonable to decide 
that a natural ordering of such transactions is that they be ordered by amount, which 
we can implement by adding an appropriate compareToO method in the class defini¬ 
tion. With such a definition, we could process an array a [] of Transacti ons by, for ex¬ 
ample, first sorting it with the call Qui ck. sort (a). Our sorting methods know nothing 
about our Transacti on data type, but Java’s Comparabl e interface allows us to define 
a natural ordering so that we can use any of our methods to sort Transacti on objects. 
Alternatively, we might specify that Transacti on objects are to be ordered by date by 
implementing compareToO to compare the Date fields. Since Date objects are them¬ 
selves Comparable, we can just invoke the compareToO method in Date rather than 
having to implement it from scratch. It is also reasonable to consider ordering this data 
by it customer field; arranging to allow clients the flexibility to switch among multiple 
different orders is an interesting challenge that we will soon consider. 


public int compareTo(Transaction that) 

{ return this.when.compareTo(that.when); } 

Alternate compareToO implementation for sorting transactions by date 
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Pointer sorting. The approach we are using is known in the classical literature as 
pointer sorting, so called because we process references to items and do not move the 
data itself. In programming languages such as C and C++, programmers explicitly de¬ 
cide whether to manipulate data or pointers to data; in Java, pointer manipulation is 
implicit. Except for primitive numeric types, we always manipulate references to ob¬ 
jects (pointers), not the objects themselves. Pointer sorting adds a level of indirection: 
the array contains references to the objects to be sorted, not the objects themselves. We 
briefly consider some associated issues, in the context of sorting. With multiple refer¬ 
ence arrays, we can have multiple different sorted representations of different parts of a 
single body of data (perhaps using multiple keys, as described below). 

Keys are immutable. It stands to reason that an array might not remain sorted if a 
client is allowed to change the values of keys after the sort. Similarly, a priority queue 
can hardly be expected to operate properly if the client can change the values of keys 
between operations. In Java, it is wise to ensure that key values do not change by using 
immutable keys. Most of the standard data types that you are likely to use as keys, such 
as Stri ng, Integer, Doubl e, and Fi 1 e, are immutable. 

Exchanges are inexpensive. Another advantage of using references is that we avoid the 
cost of moving full items. The cost saving is significant for arrays with large items (and 
small keys) because the compare needs to access just a small part of the item, and most 
of the item is not even touched during the sort. The reference approach makes the cost 
of an exchange roughly equal to the cost of a compare for general situations involving 
arbitrarily large items (at the cost of the extra space for the references). Indeed, if the 
keys are long, the exchanges might even wind up being less costly than the compare. 
One way to study the performance of algorithms that sort arrays of numbers is to sim¬ 
ply look at the total number of compares and exchanges they use, implicitly making the 
assumption that the cost of exchanges is the same as the cost of compares. Conclusions 
based on this assumption are likely to apply to a broad class of applications in Java, 
because we are sorting reference objects. 

Alternate orderings. There are many applications where we want to use differ¬ 
ent orders for the objects that we are sorting, depending on the situation. The Java 
Comparator interface allows us to build multiple orders within a single class. It has 
a single public method compare () that compares two objects. If we have a data type 
that implements this interface, we can pass a Comparator to sort() (which passes itto 
I ess O) as in the example on the next page. The Comparator mechanism allows us to 
sort arrays of any type of object, using any total order that we wish to define for them. 
Using a Comparator instead of working with Comparable types better separates the 
definition of the type from the definition of what it means to compare two objects of 
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that type. Indeed, there are typically many possible ways to compare objects, and the 
Comparator mechanism allows us to choose among them. For instance, to sort an ar¬ 
ray a [] of strings without regard to whether characters are uppercase or lowercase you 
can just call Insertion. sort(a, Stri ng. CASE_INSENSITIVE_ORDER) which makes 
use of the CASE_INSENSITIVE_ORDER comparator defined in Java’s String class. As 
you can imagine, the precise rules for ordering strings are complicated and quite differ¬ 
ent for various natural languages, so Java has many Stri ng comparators. 

Items with multiple keys. In typical applications, items have multiple instance variables 
that might need to serve as sort keys. In our transaction example, one client may need 
to sort the transaction list by customer (for example, to bring together all transactions 
involving each customer); another client might need to sort the list by amount (for 
example, to identify high-value transactions); and other clients might need to use other 
fields as sort keys. The Comparator mechanism is precisely what we need to allow this 
flexibility. We can define multiple comparators, as in the alternate implementation of 
Transacti on shown on the bottom of the next page. With this definition, a client can 
sort an array of Transacti on objects by time with the call 

Insertion.sortfa, new Transaction.WhenOrderO) 
or by amount with the call 

Insertion.sortfa, new Transaction.HowMuchOrder()). 

The sort does each compare through a callback to the compare () method in 
Transaction that is specified by the client code. To avoid the cost of making a new 
Comparator object for each sort, we could use pub! i c fi nal instance variables to de¬ 
fine the comparators (as Java does for CASE_INSENSITIVE_ORDER). 


public static void sort(Object[] a, Comparator c) 

{ 

int N = a.length; 

for (int i =1; i < N; i++) 

for (int j = i ; j > 0 && less(c, a[j], a[j-l]); j—) 
exch(a, j, j-1); 

} 

private static boolean less(Comparator c, Object v, Object w) 
{ return c.compare(v, w) < 0; } 

private static void exch(Object[] a, int i, int j) 

{ Object t = a[i] ; a[i] = a[j] ; a[j] = t; } 

Insertion sorting with a Comparator 
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Priority queues with comparators. The same flexibility to use comparators is also 
useful for priority queues. Extending our standard implementation in algorithm 2.6 
to support comparators involves the following steps: 

■ Import java. util .Comparator. 

■ Add to MaxPQ an instance variable comparator and a constructor that takes a 
comparator as argument and initializes comparator to that value. 

■ Add code to 1 ess () that checks whether comparator is nul 1 (and uses it if it is 
not nul 1). 

For example, with these changes, you could build different priority queues with 
Transaction keys, using the time, place, or account number for the ordering. If you 
remove the Key extends Comparabl e<Key> phrase from Mi nPQ, you even can support 
keys with no natural order. 


import java.util.Comparator; 

public class Transaction 

{ 

private final String who; 
private final Date when; 
private final double amount; 

public static class WhoOrder implements Comparator<Transaction> 

{ 

public int compare(Transaction v, Transaction w) 

{ return v.who.compareTofw.when); } 

} 

public static class WhenOrder implements Comparator<Transaction> 

{ 

public int compare(Transaction v, Transaction w) 

{ return v.when.compareTofw.when); } 

} 

public static class HowMuchOrder implements Comparator<Transaction> 

{ 

public int compare(Transaction v, Transaction w) 

{ 

if (v.amount < w.amount) return -1; 
if (v.amount > w.amount) return +1; 
return 0; 

} 

} 

} 

Insertion sorting with a Comparator 
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Stability. A sorting method is stable if it preserves the relative order of equal keys in the 
array. This property is frequently important. For example, consider an internet com¬ 
merce application where we have to process a large number of events that have loca¬ 
tions and timestamps. To begin, suppose that we store events in an array as they arrive, 
so they are in order of the timestamp in the array. Now suppose that the application 
requires that the transactions be separated out by location for further processing. One 
easy way to do so is to sort the array by location. If the sort is unstable, the transac¬ 
tions for each city may not necessarily be in order by timestamp after the sort. Often, 
programmers who are unfamiliar with stability are surprised, when they first encounter 
the situation, by the way an unstable algorithm seems to scramble the data. Some of 
the sorting methods that we have considered in this chapter are stable (insertion sort 
and mergesort); many are not (selection sort, shellsort, quicksort, and heapsort). There 
are ways to trick any sort into stable behavior (see exercise 2.5.18), but using a stable 
algorithm is generally preferable when stability is an essential requirement. It is easy 
to take stability for granted; actually, no practical method in common use achieves 
stability without using significant extra time or space (researchers have developed al¬ 
gorithms that do so, but applications programmers have judged them too complicated 
to be useful). 


sorted by time 


sorted by location (not stable) 


sorted by location (stable) 


Chicago 09:00:00 
Phoenix 09:00:03 
Houston 09:00:13 
Chicago 09:00:59 
Houston 09:01:10 
Chicago 09:03:13 
Seattle 09:10:11 
Seattle 09:10:25 
Phoenix 09:14:25 
Chicago 09:19:32 
Chicago 09:19:46 
Chicago 09:21:05 
Seattle 09:22:43 
Seattle 09:22:54 
Chicago 09:25:52 
Chicago 09:35:21 
Seattle 09:36:14 
Phoenix 09:37:44 


Chicago 09:25:52 
Chicago 09:03:13 
Chicago 09:21:05 
Chicago 09:19:46 
Chicago 09:19:32 
Chicago 09:00:00 
Chicago 09:35:21 
Chicago 09:00:59 
Houston 09:01:10 
Houston 09:00:13 
Phoenix 09:37:44 
Phoenix 09:00:03 
Phoenix 09:14:25 
Seattle 09:10:25 
Seattle 09:36:14 
Seattle 09:22:43 
Seattle 09:10:11 
Seattle 09:22:54 


Chicago 09:00:00 
Chicago 09:00:59 
Chicago 09:03:13 
Chicago 09:19:32 
Chicago 09:19:46 
Chicago 09:21:05 
Chicago 09:25:52 
Chicago 09:35:21 
Houston 09:00:13 
Houston 09:01:10 
Phoenix 09:00:03 
Phoenix 09:14:25 
Phoenix 09:37:44 
Seattle 09:10:11 
Seattle 09:10:25 
Seattle 09:22:43 
Seattle 09:22:54 
Seattle 09:36:14 




Stability when sorting on a second key 
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Which sorting algorithm should I use? We have considered numerous sorting 
algorithms in this chapter, so this question is natural. Knowing which algorithm is best 
possible depends heavily on details of the application and implementation, but we 
have studied some general-purpose methods that can be nearly as effective as the best 
possible for a wide variety of applications. 

The table at the bottom of this page is a general guide that summarizes the impor¬ 
tant characteristics of the sort algorithms that we have studied in this chapter. In all 
cases but shellsort (where the growth rate is only an estimate), insertion sort (where 
the growth rate depends on the order of the input keys), and both versions of quicksort 
(where the growth rate is probabilitic and may depend on the distribution of input key 
values), multiplying these growth rates by appropriate constants gives an effective way 
to predict running time. The constants involved are partly algorithm-dependent (for 
example, heapsort uses twice the number of compares as mergesort and both do many 
more array accesses than quicksort) but are primarily dependent on the implementa¬ 
tion, the Java compiler, and your computer, which determine the number of machine 
instructions that are executed and the time that each requires. Most important, since 
they are constants, you can generally predict the running time for large ;Y by running 
experiments for smaller N and extrapolating, using our standard doubling protocol. 


algorithm 

stable? 

in place? 

order of growth to sort N items 
running time extra space 

notes 

selection sort 

no 

yes 

N 2 

1 


insertion sort 

yes 

yes 

between 

N and N 2 

1 

depends on order 
of items 

shellsort 

no 

yes 

iV log iV ? 
N 615 ? 

1 


quicksort 

no 

yes 

IV log IV 

IgN 

probabilistic 

guarantee 

3-way quicksort 

no 

yes 

between 

N and N log N 

lg N 

probabilistic, 
also depends on 
distribution of 
input keys 

mergesort 

yes 

no 

N log IV 

N 


heapsort 

no 

yes 

IV log IV 

1 



Performance characteristics of sorting algorithms 


www.it-ebooks.info 



2.5 ■ Applications 


343 


Property T. Quicksort is the fastest general-purpose sort. 

Evidence: This hypothesis is supported by countless implementations of quick¬ 
sort on countless computer systems since its invention decades ago. Generally, the 
reason that quicksort is fastest is that it has only a few instructions in its inner 
loop (and it does well with cache memories because it most often references data 
sequentially) so that its running time is ~c Nig N with the value of c smaller than 
the corresponding constants for other linearithmic sorts. With 3 -way partitioning, 
quicksort becomes linear for certain key distributions likely to arise in practice, 
where other sorts are linearithmic. 


Thus, in most practical situations, quicksort is the method of choice. Still, given the 
broad reach of sorting and the broad variety of computers and systems, a flat statement 
like this is difficult to justify. For example, we have already seen one notable exception: 
if stability is important and space is available, mergesort might be best. We will see oth¬ 
er exceptions in chapter 5. With tools like SortCompare and a considerable amount of 
time and effort, you can do a more detailed study of comparative performance of these 
algorithms and the refinements that we have discussed for your computer, as discussed 
in several exercises at the end of this section. Perhaps the best way to interpret prop¬ 
erty t is as saying that you certainly should seriously consider using quicksort in any 
sort application where running time is important. 

Sorting primitive types. In some performance-critical applications, the focus may be 
on sorting numbers, so it is reasonable to avoid the costs of using references and sort 
primitive types instead. For example, consider the difference between sorting an array 
of doubl e values and sorting an array of Doubl e values. In the former case, we exchange 
the numbers themselves and put them in order in the array; in the latter, we exchange 
references to Doubl e objects, which contain the numbers. If we are doing nothing more 
than sorting a huge array of numbers, we avoid paying the cost of storing an equal 
number of references plus the extra cost of accessing the numbers through the refer¬ 
ences, not to mention the cost of invoking compareToO and 1 ess() methods. We can 
develop efficient versions of our sort codes for such purposes by replacing Comparabl e 
with the primitive type name, and redefining 1 ess () or just replacing calls to 1 ess 0 
with code like a [i ] < a[j] (see exercise 2.1.26). 

Java system sort. As an example of applying the information given in the table on 
page 342, consider Java’s primary system sort method, java.util .Arrays.sort(). 
With overloading of argument types, this name actually represents a collection of 
methods: 
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■ A different method for each primitive type 

■ A method for data types that implement Comparabl e 

■ A method that uses a Comparator 

Java’s systems programmers have chosen to use quicksort (with 3 -way partitioning) 
to implement the primitive-type methods, and mergesort for reference-type methods. 
The primary practical implications of these choices are, as just discussed, to trade speed 
and memory usage (for primitive types) for stability (for reference types). 

the algorithms and ideas that we have been considering are an essential part of 
many modern systems, including Java. When developing Java programs to address an 
application, you are likely to find that Java’s Arrays. sort O implementations (perhaps 
supplemented by your own implementation(s) of compareToO and/or compareO) 
will meet your needs, because you will be using 3 -way quicksort or mergesort, both 
proven classic algorithms. 

In this book, we generally will use our own Qui ck. sort() (usually) or Merge. sort () 
(when stability is important and space is not) in sort clients. You may feel free to use 
Arrays. sortO unless you have a good reason to use another specific method. 

Reductions The idea that we can use sorting algorithms to solve other problems 
is an example of a basic technique in algorithm design known as reduction. We con¬ 
sider reduction in detail in chapter 6 because of its importance in the theory of al¬ 
gorithms—in the meantime, we will consider several practical examples. A reduction 
is a situation where an algorithm developed for one problem is used to solve another. 
Applications programmers are quite used to the concept of reduction (whether or not 
it is explicitly articulated)—every time you make use of a method that solves problem 
B in order to solve problem A, you are doing a reduction from A to B. Indeed, one goal 
in implementing algorithms is to facilitate reductions by making the algorithms useful 
for as wide a variety as possible of applications. We begin with a few elementary exam¬ 
ples for sorting. Many of these take the form of algorithmic puzzles where a quadratic 
brute-force algorithm is immediate. It is often the case that sorting the data first makes 
it easy to finish solving the problem in linear additional time, thus reducing the total 
cost from quadratic to linearithmic. 

Duplicates. Are there any duplicate keys in an array of Comparable objects? How 
many distinct keys are there? Which value appears most frequently? For small arrays, 
these kinds of questions are easy to answer with a quadratic algorithm that compares 
each array entry with each other array entry. For large arrays, using a quadratic algo¬ 
rithm is not feasible. With sorting, you can answer these questions in linearithmic time: 
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first sort the array, then make a pass through the sorted array, taking note of duplicate 
keys that appear consecutively in the ordered array. For example, the code fragment at 
right counts the distinct keys in an array. With simple modifications to this code, you 
can answer the questions above and perform tasks such as printing all the distinct val¬ 
ues, all the values that are duplicated, and so forth, even for huge arrays. 

Rankings. A permutation (or ranking ) is an 
array of N integers where each of the integers 
between 0 and N-l appears exactly once. The 
Kendall tau distance between two rankings is 
the number of pairs that are in different order 
in the two rankings. For example, the Kendall 
tau distance between 0 3 1 6 2 5 4 and 
1 0 3 6 4 2 5 is four because the pairs 0-1, 

3-1, 2-4, 5-4 are in different relative order in 
the two rankings, but all other pairs are in the same relative order. This statistic is widely 
used: in sociology to study social choice and voting theory, in molecular biology to 
compare genes using expression profiles, and in ranking search engine results on the 
web, among many other applications. The Kendall tau distance between a permutation 
and the identity permutation (where each entry is equal to its index) is the number 
of inversions in the permutation, and a quadratic algorithm based on insertion sort 
is not difficult to devise (recall proposition c in section 2 . 1 ). Efficiently computing 
the Kendall tau distance is an interesting exercise for a programmer (or a student!) 
who is familiar with the classical sorting algorithms that we have studied (see exercise 
2 . 5 . 1 . 9 ). 

Priority-queue reductions. In section 2 . 4 , we considered two examples of problems 
that reduce to a sequence of operations on priority queues. To pM, on page 311, finds the M 
items in an input stream with the highest keys. Mul ti way, on page 322, merges M sorted 
input streams together to make a sorted output stream. Both of these problems are eas¬ 
ily addressed with a priority queue of size M. 

Median and order statistics. An important application related to sorting but for 
which a full sort is not required is the operation of finding the median of a collection 
of keys (the value with the property that half the keys are no larger and half the keys 
are no smaller). This operation is a common computation in statistics and in various 
other data-processing applications. Finding the median is a special case of selection: 
finding the /cth smallest of a collection of numbers. Selection has many applications in 
the processing of experimental and other data. The use of the median and other order 


Quick, sort(a) ; 

int count = 1; // Assume a.length > 0. 

for (int i = 1; i < a.length; i++) 
if (a[i] . compareTo(a[i-1]) != 0) 
count++; 

Counting the distinct keys in a[] 
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statistics to divide an array into smaller groups is common. Often, only a small part of a 
large array is to be saved for further processing; in such cases, a program that can select, 
say, the top 10 percent of the items of the array might be more appropriate than a full 
sort. Our TopM application of section 
2.4 solves this problem for an unbound¬ 
ed input stream, using a priority queue. 

An effective alternative to TopM when 
you have the items in an array is to just 
sort it: after the call Quick.sort(a) the k 
smallest values in the array are in the first 
k array positions for all k less than the ar¬ 
ray length. But this approach involves a 
sort, so the running time is linearithmic. 

Can we do better? Finding the k smallest 
values in an array is easy when k is very 
small or very 


public static Comparable 
select(Comparable[] a, int 
{ 

StdRandom.shuffle(a); 
int lo = 0, hi = a.length 
while (hi > lo) 

{ 

int j = partition (a, lo 
if (j == k) return 


el se 
el se 


if 

if 


} 

return a[k] ; 
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Partitioning to find the median 


large, but more Selecting the k smallest elements in a[] 
challenging 

when k is a constant fraction of the array size, such as finding 
the median ( k= M2). You might be surprised to learn that it is 
possible to solve this problem in linear time, as in the sel ect () 
method above (this implementation requires a client cast; for 
the more pedantic code needed to avoid this requirement, see 
the booksite). To do the job, sel ect() maintains the variables 
lo and hi to delimit the subarray that contains the index k 
of the item to be selected and uses quicksort partitioning to 
shrink the size of the subarray. Recall that parti ti on O rear¬ 
ranges an array a[lo] through a [hi] and returns an integer 
j such that a[lo] through a[j-l] are less than or equal to 
a[j], and a[j+l] through a[hi] are greater than or equal to 
a[ j] . Now, if k is equal to j, then we are done. Otherwise, if 
k < j , then we need to continue working in the left subarray 
(by changing the value of hi to j—1); if k > j, then we need 
to continue working in the right subarray (by changing 1 o to 
j+1). The loop maintains the invariant that no entry to the left 
of lo is larger and no entry to the right of hi is smaller than 
any element within a [1 o . . hi ]. After partitioning, we preserve 
this invariant and shrink the interval until it consists just of 
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k. Upon termination, a[k] contains the (k + l)st smallest entry, a[0] through a[k-l] 
are all smaller than (or equal to) a [k], and a [k+1] through the end of the array are all 
greater than (or equal to) a[k]. To gain some insight into why this is a linear-time al¬ 
gorithm, suppose that partitioning divides the array exactly in half each time. Then the 
number of compares is N + N/2 + N /4 + N/8 + ... , terminating when the kth small¬ 
est item is found. This sum is less than 2 N. As with quicksort, it takes a bit of math to 
find the true bound, which is a bit higher. Also as with quicksort, the analysis depends 
on partitioning on a random item, so that the guarantee is probabilistic. 

Proposition U. Partitioning-based selection is a linear-time algorithm, on average. 

Proof: An analysis similar to, but significantly more complex than, the proof of 
proposition k for quicksort leads to the result that the average number of com¬ 
pares is ~ 2 N + 2k\n{N/k) + 2 (N — k) In (N/(N — k)), which is linear for any 
allowed value of k. For example, this formula says that finding the median ( k = 
N/2) requires ~ (2 + 2 In 2 )N compares, on the average. Note that the worst case 
is quadratic but randomization protects against that possibility, as with quicksort. 


Designing a selection algorithm that is guaranteed to use a linear number of compares 
in the worst case is a classic result in computational complexity, but it has not yet led to 
a useful practical algorithm. 
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A brief survey of sorting applications Direct applications of sorting are fa¬ 
miliar, ubiquitous, and far too numerous for us to list them all. You sort your music by 
song title or by artist name, your email or phone calls by time or origin, and your pho¬ 
tos by date. Universities sort student accounts by name or ID. Credit card companies 
sort millions or even billions of transactions by date or amount. Scientists sort not only 
experimental data by time or other identifier but also to enable detailed simulations 
of the natural world, from the motion of particles or heavenly bodies to the structure 
of materials to social interations and relationships. Indeed, it is difficult to identify a 
computational application that does not involve sorting! To elaborate upon this point, 
we describe in this section examples of applications that are more complicated than the 
reductions just considered, including several that we will examine in detail later in this 
book. 

Commercial computing. The world is awash in information. Government organiza¬ 
tions, financial institutions, and commercial enterprises organize much of this infor¬ 
mation by sorting it. Whether the information is accounts to be sorted by name or 
number, transactions to be sorted by date or amount, mail to be sorted by postal code 
or address, files to be sorted by name or date, or whatever, processing such data is sure 
to involve a sorting algorithm somewhere along the way. Typically, such information 
is organized in huge databases, sorted by multiple keys for efficient search. An effective 
strategy that is widely used is to collect new information, add it to the database, sort it 
on the keys of interest, and merge the sorted result for each key into the existing data¬ 
base. The methods that we have discussed have been used effectively since the early days 
of computing to build a huge infrastructure of sorted data and methods for processing 
it that serve as the basis for all of this commercial activity. Arrays having millions or 
even billions of entries are routinely processed today—without linearithmic sorting 
algorithms, such arrays could not be sorted, making such processing extremely difficult 
or impossible. 

Search for information. Keeping data in sorted order makes it possible to efficiently 
search through it using the classic binary search algorithm (see chapter i). You will 
also see that the same scheme makes it easy to quickly handle many other kinds of 
queries. How many items are smaller than a given item? Which items fall within a given 
range? In chapter 3 , we consider such questions. We also consider in detail various 
extensions to sorting and binary search that allow us to intermix such queries with 
operations that insert and remove objects from the set, still guaranteeing logarithmic 
performance for all operations. 
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Operations research. The field of operations research (OR) develops and applies math¬ 
ematical models for problem-solving and decision-making. We will see several exam¬ 
ples in this book of relationships between OR and the study of algorithms, beginning 
here with the use of sorting in a classic OR problem known as scheduling. Suppose that 
we have N jobs to complete, where job j requires t } seconds of processing time. We need 
to complete all of the jobs but want to maximize customer satisfaction by minimizing 
the average completion time of the jobs. The shortest processing time first rule, where we 
schedule the jobs in increasing order of processing time, is known to accomplish this 
goal. Therefore we can sort the jobs by processing time or put them on a minimum- 
oriented priority queue. With various other constraints and restrictions, we get various 
other scheduling problems, which frequently arise in industrial applications and are 
well-studied. As another example, consider the load-balancing problem, where we have 
M identical processors and N jobs to complete, and our goal is to schedule all of the 
jobs on the processors so that the time at which the last job completes is as early as pos¬ 
sible. This specific problem is NP-hard (see chapter 6) so we do not expect to find a 
practical way to compute an optimal schedule. One method that is known to produce 
a good schedule is the longest processing time first rule, where we consider the jobs in 
descending order of processing time, assigning each job to the processor that becomes 
available first. To implement this algorithm, we first sort the jobs in reverse order. Then 
we maintain a priority queue of M processors, where the priority is the sum of the pro¬ 
cessing times of its jobs. At each step, we delete the processor with the minimum prior¬ 
ity, add the next job to the processor, and reinsert that processor into the priority queue. 

Event-driven simulation. Many scientific applications involve simulation, where the 
point of the computation is to model some aspect of the real world in order to be able to 
better understand it. Before the advent of computing, scientists had little choice but to 
build mathematical models for this purpose; such models are now well-complemented 
by computational models. Doing such simulations efficiently can be challenging, and 
use of appropriate algorithms certainly can make the difference between being able 
to complete the simulation in a reasonable amount of time and being stuck with the 
choice of accepting inaccurate results or waiting for the simulation to do the computa¬ 
tion necessary to get accurate results. We will consider in chapter 6 a detailed example 
that illustrates this point. 

Numerical computations. Scientific computing is often concerned with accuracy 
(how close are we to the true answer?). Accuracy is extremely important when we are 
performing millions of computations with estimated values such as the floating-point 
representation of real numbers that we commonly use on computers. Some numeri¬ 
cal algorithms use priority queues and sorting to control accuracy in calculations. For 
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example, one way to do numerical integration (quadrature), where the goal is to esti¬ 
mate the area under a curve, is to maintain a priority queue with accuracy estimates for 
a set of subintervals that comprise the whole interval. The process is to remove the least 
accurate subinterval, split it in half (thus achieving better accuracy for the two halves), 
and put the two halves back onto the priority queue, continuing until a desired toler¬ 
ance is reached. 

Combinatorial search. A classic paradigm in artificial intelligence and in coping with 
intractable problems is to define a set of configurations with well-defined moves from 
one configuration to the next and a priority associated with each move. Also defined 
is a start configuration and a goal configuration (which corresponds to having solved 
the problem). The well-known A* algorithm is a problem-solving process where we put 
the start configuration on the priority queue, then do the following until reaching the 
goal: remove the highest-priority configuration and add to the queue all configurations 
that can be reached from that with one move (excluding the one just removed). As with 
event-driven simulation, this process is tailor-made for priority queues. It reduces solv¬ 
ing the problem to defining an effective priority function. See exercise 2 . 5.32 for an 
example. 

beyond such direct applications (and we have only indicated a small fraction of 
those), sorting and priority queues are an essential abstraction in algorithm design, 
so they will surface frequently throughout this book. We next list some examples of 
applications from later in the book. All of these applications depend upon the efficient 
implementations of sorting algorithms and the priority-queue data type that we have 
considered in this chapter. 

Prim’s algorithm and Dijkstra’s algorithm are classical algorithms from chapter 4 . 
That chapter is about algorithms that process graphs, a fundamental model for items 
and edges that connect pairs of items. The basis for these and several other algorithms 
is graph search, where we proceed from item to item along edges. Priority queues play a 
fundamental role in organizing graph searches, enabling efficient algorithms. 

Kruskal’s algorithm is another classic algorithm for graphs whose edges have weights 
that depends upon processing the edges in order of their weight. Its running time is 
dominated by the cost of the sort. 

Huffman compression is a classic data compression algorithm that depends upon pro¬ 
cessing a set of items with integer weights by combining the two smallest to produce 
a new one whose weight is the sum of its two constituents. Implementing this opera- 
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tion is immediate, using a priority queue. Several other data-compression schemes are 
based upon sorting. 

String-processing algorithms, which are of critical importance in modern applica¬ 
tions in cryptology and in genomics, are often based on sorting (generally using one 
of the specialized string sorts discussed in chapter 5 ). For example, we will discuss in 
chapter 6 algorithms for finding the longest repeated substring in a given string that is 
based on first sorting suffixes of the strings. 
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Q&A 

Q. Is there a priority-queue data type in the Java library? 
A. Yes, see java, uti 1 . PriorityQueue. 
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EXERCISES 


2 . 5.1 Consider the following implementation of the compareToO method for Stri ng. 
How does the third line help with efficiency? 

public int compareTo(String that) 

{ 

if (this == that) return 0; // this line 

int n = Math.min(this.length() , that.lengthO); 
for (int i =0; i < n; i++) 

{ 

if (this.charAt(i) < that.charAt(i)) return -1; 

else if (this.charAt(i) > that.charAt(i)) return +1; 

} 

return thi s . 1 ength() - that.lengthO; 

} 

2 . 5.2 Write a program that reads a list of words from standard input and prints all two- 
word compound words in the list. For example, if after, thought, and afterthought 
are in the list, then afterthought is a compound word. 

2 . 5.3 Criticize the following implementation of a class intended to represent account 
balances. Why is compareToO a flawed implementation of the Comparabl e interface? 

public class Balance implements Comparable<Balance> 

{ 

private double amount; 

public int compareTo(Balance that) 

{ 

if (this.amount < that.amount - 0.005) return -1; 
if (this.amount > that.amount + 0.005) return +1; 
return 0; 

} 

} 

Describe a way to fix this problem. 

2 . 5.4 Implement a method Stri ng [] dedup(Stri ng [] a) that returns the objects in 
a [] in sorted order, with duplicates removed. 

2 . 5.5 Explain why selection sort is not stable. 
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EXERCISES (continued) 


2 . 5.6 Implement a recursive version of sel ect(). 

2 . 5.7 About how many compares are required, on the average, to find the smallest of 
N items using sel ectO? 

2 . 5.8 Write a program Frequency that reads strings from standard input and prints 
the number of times each string occurs, in descending order of frequency. 

2 . 5.9 Develop a data type that allows you to write a client that can sort a file such as the 
one shown at right. 

2 . 5.10 Create a data type Version that represents a 
software version number, such as 115.1.1,115.10.1, 

115.10.2. Implement the Comparable interface so 
that 115.1.1 is less than 115.10.1, and so forth. 

2 . 5 . 11 One way to describe the result of a sorting al¬ 
gorithm is to specify a permutation p[] of the num¬ 
bers 0 to a.length-1, such that p[i] specifies where 
the key originally in a [i ] ends up. Give the permuta¬ 
tions that describe the results of insertion sort, selec¬ 
tion sort, shellsort, mergesort, quicksort, and heapsort 
for an array of seven equal keys. 


input (DJI volumes for each day) 


1- 0ct-28 

2- Oct-28 
B-Oct-28 

4- 0ct-28 

5- Oct-28 


B500000 

3850000 

4060000 

4330000 

4360000 


30- Dec-99 554680000 

31- Dec-99 374049984 

3- Jan-00 931800000 

4- ]an-00 1009000000 

5- Jan-00 1085500032 

output 

19-Aug-40 130000 
26-Aug-40 160000 
24-lul-40 200000 
10-Aug-42 210000 
23-lun-42 210000 


23- lul-02 2441019904 
17-Jul-02 2566500096 
15-lul-02 2574799872 
19-lul-02 2654099968 

24- lul-02 2775559936 
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CREATIVE PROBLEMS 


2.5.12 Scheduling. Write a program SPT. java that reads job names and processing 
times from standard input and prints a schedule that minimizes average completion 
time using the shortest processing time first rule, as described on page 349. 

2.5.13 Load balancing. Write a program LPT. java that takes an integer M as a com¬ 
mand-line argument, reads job names and processing times from standard input and 
prints a schedule assigning the jobs to M processors that approximately minimizes the 
time when the last job completes using the longest processing time first rule, as de¬ 
scribed on page 349. 

2.5.14 Sort by reverse domain. Write a data type Domai n that represents domain names, 
including an appropriate compareToO method where the natural order is in order of 
the reverse domain name. For example, the reverse domain of cs.princeton.edu is 
edu. princeton. cs. This is useful for web log analysis. Hint. Use s. split ("\\. ") to 
split the string s into tokens, delimited by dots. Write a client that reads domain names 
from standard input and prints the reverse domains in sorted order. 

2.5.15 Spam campaign. To initiate an illegal spam campaign, you have a list of email 
addresses from various domains (the part of the email address that follows the @ 
symbol). To better forge the return addresses, you want to send the email from an¬ 
other user at the same domain. For example, you might want to forge an email from 
wayne@pri nceton. edu to rs@pri nceton . edu. How would you process the email list 
to make this an efficient task? 

2.5.16 Unbiased election. In order to thwart bias against candidates whose names ap¬ 
pear toward the end of the alphabet, California sorted the candidates appearing on its 
2003 gubernatorial ballot by using the following order of characters: 

RWQOJMVAHBSGZXNTCIEKUPDYFL 

Create a data type where this is the natural order and write a client Cal i form’ a with a 
single static method mai n Q that sorts strings according to this ordering. Assume that 
each string is composed solely of uppercase letters. 

2.5.17 Check stability. Extend your check() method from exercise 2 . 1.16 to call 
sortO for a given array and return true if sort() sorts the array in order in a stable 
manner, fal se otherwise. Do not assume that sortO is restricted to move data only 
with exch(). 
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CREATIVE PROBLEMS (continued) 


2.5.18 Force stability. Write a wrapper method that makes any sort stable by creating 
a new key type that allows you to append each key’s index to the key, call sortO, then 
restore the original key after the sort. 

2.5.19 Kendall tau distance. Write a program KendallTau. java that computes the 
Kendall tau distance between two permutations in linearithmic time. 

2.5.20 Idle time. Suppose that a parallel machine processes N jobs. Write a program 
that, given the list of job start and finish times, finds the largest interval where the ma¬ 
chine is idle and the largest interval where the machine is not idle. 

2.5.21 Multidimensional sort. Write a Vector data type for use in having the sort¬ 
ing methods sort multidimensional vectors of d integers, putting the vectors in order 
by first component, those with equal first component in order by second component, 
those with equal first and second components in order by third component, and so 
forth. 


2.5.22 Stock market trading. Investors place buy and sell orders for a particular stock 
on an electronic exchange, specifying a maximum buy or minimum sell price that they 
are willing to pay, and how many shares they wish to trade at that price. Develop a 
program that uses priority queues to match up buyers and sellers and test it through 
simulation. Maintain two priority queues, one for buyers and one for sellers, executing 
trades whenever a new order can be matched with an existing order or orders. 

2.5.23 Sampling for selection. Investigate the idea of using sampling to improve selec¬ 
tion. Hint: Using the median may not always be helpful. 

2.5.24 Stable priority queue. Develop a stable priority-queue implementation (which 
returns duplicate keys in the same order in which they were inserted). 

2.5.25 Points in the plane. Write three stati c comparators for the Poi nt2D data type 
of page 77, one that compares points by their x coordinate, one that compares them by 
their y coordinate, and one that compares them by their distance from the origin. Write 
two non-stati c comparators for the Poi nt2D data type, one that compares them by 
their distance to a specified point and one that compares them by their polar angle with 
respect to a specified point. 

2.5.26 Simple polygon. Given N points in the plane, draw a simple polygon with N 
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points as vertices. Hint: Find the point p with the smallest y coordinate, breaking ties 
with the smallest x coordinate. Connect the points in increasing order of the polar angle 
they make with p. 

2.5.27 Sorting parallel arrays. When sorting parallel arrays, it is useful to have a version 
of a sorting routine that returns a permutation, say i ndex[], of the indices in sorted 
order. Add a method i ndi rectSortO to Insertion that takes an array of Comparabl e 
objects a [] as argument, but instead of rearranging the entries of a [] returns an integer 
array index [] sothata[index[0]] through a [index [N-l]] are the items in ascending 
order. 


2.5.28 Sort files by name. Write a program FileSorter that takes the name of a 
directory as a command-line argument and prints out all of the files in the current 
directory, sorted by file name. Hint: Use the Fi 1 e data type. 

2.5.29 Sort files by size and date of last modification. Write comparators for the type 
File to order by increasing/decreasing order of file size, ascending/descending order 
of file name, and ascending/descending order of last modification date. Use these 
comparators in a program LS that takes a command-line argument and lists the files 
in the current directory according to a specified order, e.g., " -t" to sort by timestamp. 
Support multiple flags to break ties. Be sure to use a stable sort. 

2.5.30 Boerners theorem. True or false: If you sort each column of a matrix, then sort 
each row, the columns are still sorted. Justify your answer. 
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| EXPERIMENTS 


2.5.31 Duplicates. Write a client that takes integers M, N, and T as command-line argu¬ 
ments, then uses the code given in the text to perform T trials of the following experi¬ 
ment: Generate N random i nt values between 0 and M - 1 and count the number of 
duplicates. Run your program for T = 10 and N= 10 3 , 10 4 , 10 5 , and 10 6 , with M = N/ 2, 
and N, and 2 N. Probability theory says that the number of duplicates should be about 
(1 — e““) where a = N/M —print a table to help you confirm that your experiments 
validate that formula. 

2.5.32 8 puzzle. The 8 puzzle is a game popularized by S. Loyd in the 1870s. It is 
played on a 3-by-3 grid with 8 tiles labeled 1 through 8 and a blank square. Your goal 
is to rearrange the tiles so that they are in order. You are permitted to slide one of the 
available tiles horizontally or vertically (but not diagonally) into the blank square. Write 
a program that solves the puzzle using the A* algorithm. Start by using as priority the 
sum of the number of moves made to get to this board position plus the number of tiles 
in the wrong position. (Note that the number of moves you must make from a given 
board position is at least as big as the number of tiles in the wrong place.) Investigate 
substituting other functions for the number of tiles in the wrong position, such as the 
sum of the Manhattan distance from each tile to its correct position, or the sums of the 
squares of these distances. 

2.5.33 Random transactions. Develop a generator that takes an argument N, generates 
N random Transaction objects (see exercises 2.1.21 and 2.1.22), using assumptions 
about the transactions that you can defend. Then compare the performance of shellsort, 
mergesort, quicksort, and heapsort for sorting N transactions, for iV=10 3 , 10 4 , 10 5 , and 
10 6 . 
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M odern computing and the internet have made accessible a vast amount of 
information. The ability to efficiently search through this information is 
fundamental to processing it. This chapter describes classical searching algo¬ 
rithms that have proven to be effective in numerous diverse applications for decades. 
Without algorithms like these, the development of the computational infrastructure 
that we enjoy in the modern world would not have been possible. 

We use the term symbol table to describe an abstract mechanism where we save in¬ 
formation (a value ) that we can later search for and retrieve by specifying a key. The 
nature of the keys and the values depends upon the application. There can be a huge 
number of keys and a huge amount of information, so implementing an efficient sym¬ 
bol table is a significant computational challenge. 

Symbol tables are sometimes called dictionaries, by analogy with the time-honored 
system of providing definitions for words by listing them alphabetically in a reference 
book. In an English-language dictionary, a key is a word and its values is the entry 
associated with the word that contains the definition, pronunciation, and etymology. 
Symbol tables are also sometimes called indices, by analogy with another time-honored 
system of providing access to terms by listing them alphabetically at the end of a book 
such as a textbook. In a book index, a key is a term of interest and its value is the list of 
page numbers that tell readers where to find that term in the book. 

After describing the basic APIs and two fundamental implementations, we consider 
three classic data structures that can support efficient symbol-table implementations: 
binary search trees, red-black trees, and hash tables. We conclude with several exten¬ 
sions and applications, many of which would not be feasible without the efficient algo¬ 
rithms that you will learn about in this chapter. 
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The primary purpose of a symbol table is to associate a value with a key. The client can 
insert key-value pairs into the symbol table with the expectation of later being able to 
search for the value associated with a given key, from among all of the key-value pairs 
that have been put into the table. This chapter describes several ways to structure this 
data so as to make efficient not just the insert and search operations, but several other 
convenient operations as well. To implement a symbol table, we need to define an un¬ 
derlying data structure and then specify algorithms for insert, search, and other opera¬ 
tions that create and manipulate the data structure. 

Search is so important to so many computer applications that symbol tables are 
available as high-level abstractions in many programming environments, including 
Java—we shall discuss Java’s symbol-table implementations in section 3 . 5 . The table 
below gives some examples of keys and values that you might use in typical applica¬ 
tions. We consider some illustrative reference clients soon, and section 3.5 is devoted 
to showing you how to use symbol tables effectively in your own clients. We also use 
symbol tables in developing other algorithms throughout the book. 

Definition. A symbol table is a data structure for key-value pairs that supports two 
operations: insert (put) a new pair into the table and search for (get) the value as¬ 
sociated with a given key. 


application 

dictionary 
book index 
file share 

account management 
web search 
compiler 


purpose of search 


key 


find definition 
find relevant pages 
find song to download 
process transactions 
find relevant web pages 
find type and value 


word 

term 

name of song 
account number 
keyword 
variable name 


Typical symbol-table applications 


value 

definition 

list of page numbers 
computer ID 
transaction details 
list of page names 
type and value 
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API The symbol table is a prototypical abstract data type (see chapter i): it repre¬ 
sents a well-defined set of values and operations on those values, enabling us to develop 
clients and implementations separately. As usual, we precisely define the operations 
by specifying an applications programming interface (API) that provides the contract 
between client and implementation: 


public class ST<Key, Value> 


voi d 

Val ue 

voi d 
boolean 
boolean 
int 

Iterable<Key> 


ST C) 

put(Key key, Value val) 

get(Key key) 

delete(Key key) 
contains(Key key) 
i sEmptyO 
size() 
keys() 


create a symbol table 

put key-value pair into the table 
(remove key from table if value is nul 1) 

value paired with key 
(null if key is absent) 

remove key (and its value) from table 
is there a value paired with key? 
is the table empty? 

number of key-value pairs in the table 
all the keys in the table 


API for a generic basic symbol table 


Before examining client code, we consider several design choices for our implementa¬ 
tions to make our code consistent, compact, and useful. 

Generics. As we did with sorting, we will consider the methods without specifying the 
types of the items being processed, using generics. For symbol tables, we emphasize the 
separate roles played by keys and values in search by specifying the key and value types 
explicitly instead of viewing keys as implicit in items as we did for priority queues in 
section 2.4. After we have considered some of the characteristics of this basic API (for 
example, note that there is no mention of order among the keys), we will consider an 
extension for the typical case when keys are Comparabl e, which enables numerous ad¬ 
ditional methods. 

Duplicate keys. We adopt the following conventions in all of our implementations: 

■ Only one value is associated with each key (no duplicate keys in a table). 

■ When a client puts a key-value pair into a table already containing that key (and 
an associated value), the new value replaces the old one. 

These conventions define the associative array abstraction, where you can think of a 
symbol table as being just like an array, where keys are indices and values are array 
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entries. In a conventional array, keys are integer indices that we use to quickly access ar¬ 
ray values; in an associative array (symbol table), keys are of arbitrary type, but we can 
still use them to quickly access values. Some programming languages (not Java) provide 
special support that allows programmers to use code such as st [key] for st. get (key) 
andst[key] = val for st. putfkey, val ) where key and val are objects of arbitrary 
type. 

Null keys. Keys must not be nul 1 . As with many mechanisms in java, use of a null key 
results in an exception at runtime (see the third Q&A on page 387). 

Null values. We also adopt the convention that no key can be associated with the value 
nul 1 . This convention is directly tied to our specification in the API that get() should 
return null for keys not in the table, effectively associating the value null with every 
key not in the table. This convention has two (intended) consequences: First, we can 
test whether or not the symbol table defines a value associated with a given key by test¬ 
ing whether get() returns null. Second, we can use the operation of calling put() 
with null as its second (value) argument to implement deletion, as described in the 
next paragraph. 

Deletion. Deletion in symbol tables generally involves one of two strategies: lazy dele¬ 
tion, where we associate keys in the table with null, then perhaps remove all such keys 
at some later time; and eager deletion, where we remove the key from the table imme¬ 
diately. As just discussed, the code put (key, null) is an easy (lazy) implementation 
of del ete (key). When we give an (eager) implementation of del ete(), it is intended 
to replace this default. In our symbol-table implementations that do not use the default 
del ete(), the put() implementations on the booksite begin with the defensive code 

if (val == null) { delete(key); return; } 

to ensure that no key in the table is associated with nul 1 . For economy, we do not in¬ 
clude this code in the book (and we do not call put () with a null value in client code). 

Shorthand methods. For clarity in client code, we include the methods contai ns() 
and is Empty () in the API, with the default one-line implementations shown here. 
For economy, we do not 

repeat this code, but we method default implementation 

assume it to be present void delete(Key key) put(key, null); 

in all implementations of 

the symbol-table API and boolean contains(key) return get(key) != null ; 

use these methods freely boolean isEmptyO return size() == 0; 

in client code. 

Default implementations 
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Iteration. To enable clients to process all the keys and values in the table, we might 
add the phrase implements Iterabl e<Key> to the first line of the API to specify that 
every implementation must implement an i te rato r () method that returns an iterator 
having appropriate implementations of hasNextO and next(), as described for stacks 
and queues in section 1 . 3 . For symbol tables, we adopt a simpler alternative approach, 
where we specify a keys() method that returns an Iterable<Key> object for clients 
to use to iterate through the keys. Our reason for doing so is to maintain consistency 
with methods that we will define for ordered symbol tables that allow clients to iterate 
through a specified subset of keys in the table. 

Key equality. Determining whether or not a given key is in a symbol table is based on 
the concept of object equality , which we discussed at length in section 1.2 (see page 
102). Java’s convention that all objects inherit an equal s () method and its implementa¬ 
tion of equals() both for standard types such as Integer, Double, and String and 
for more complicated types such as File and URL is a head start—when using these 
types of data, you can just use the built-in implementation. For example, if x and y are 
Stri ng values, then x. equal s (y) is true if and only if x and y have the same length 
and are identical in each character position. For such client-defined keys, you need to 
override equal s(), as discussed in section 1.2. You can use our implementation of 
equal s() for Date (page 103) as a template to develop equal s() for a type of your own. 
As discussed for priority queues on page 320, a best practice is to make Key types immu¬ 
table, because consistency cannot otherwise be guaranteed. 
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Ordered symbol tables In typical applications, keys are Comparabl e objects, so 
the option exists of using the code a.compareTo(b) to compare two keys a and b. 
Several symbol-table implementations take advantage of order among the keys that is 
implied by Comparable to provide efficient implementations of the put() and get() 
operations. More important, in such implementations, we can think of the symbol ta¬ 
ble as keeping the keys in order and consider a significantly expanded API that defines 
numerous natural and useful operations involving relative key order. For example, sup¬ 
pose that your keys are times of the day. You might be interested in knowing the earliest 
or the latest time, the set of keys that fall between two given times, and so forth. In most 
cases, such operations are not difficult to implement with the same data structures and 
methods underlying the put() and get() implementations. Specifically, for applica¬ 
tions where keys are Comparabl e, we implement in this chapter the following API: 


public class ST<Key extends Comparable<Key>, Value> 



ST () 

create an ordered symbol table 

voi d 

put(Key key, Value val) 

put key-value pair into the table 
(remove key from table if value is nul 1 ) 

Val ue 

get(Key key) 

value paired with key 
(null if key is absent) 

voi d 

deletefKey key) 

remove key (and its value) from table 

boolean 

contains(Key key) 

is there a value paired with key? 

boolean 

i sEmptyO 

is the table empty? 

int 

size() 

number of key-value pairs 

Key 

mi n() 

smallest key 

Key 

max() 

largest key 

Key 

floor(Key key) 

largest key less than or equal to key 

Key 

ceiling(Key key) 

smallest key greater than or equal to key 

int 

rank(Key key) 

number of keys less than key 

Key 

select(int k) 

key of rank k 

voi d 

deleteMinO 

delete smallest key 

voi d 

deleteMax() 

delete largest key 

int 

size(Key lo, Key hi) 

number of keys in [1 o. . hi ] 

Iterable<Key> 

keys(Key lo, Key hi) 

keys in [1 o. .hi], in sorted order 

Iterable<Key> 

keys() 

all keys in the table, in sorted order 


API for a generic ordered symbol table 
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Your signal that one of our programs is implementing this API is the presence of the 
Key extends Comparable<Key> generic type variable in the class declaration, which 
specifies that the code depends upon the keys being Comparable and implements the 
richer set of operations. Together, these operations define for client programs an or¬ 
dered symbol table. 

Minimum and maximum. Perhaps the most natural queries for a set of ordered keys 
are to ask for the smallest and largest keys. We have already encountered these opera¬ 
tions, in our discussion of priority queues in section 2.4. In ordered symbol tables, 


we also have methods to delete the maximum 
and minimum keys (and their associated val¬ 
ues). With this capability, the symbol table can 
operate like the IndexMinPQO class that we 
discussed in section 2 . 4 . The primary differ¬ 
ences are that equal keys are allowed in prior¬ 
ity queues but not in symbol tables and that 
ordered symbol tables support a much larger 
set of operations. 

Floor and ceiling. Given a key, it is often use¬ 
ful to be able to perform the floor operation 
(find the largest key that is less than or equal to 
the given key) and the ceiling operation (find 
the smallest key that is greater than or equal to 
the given key). The nomenclature comes from 
functions defined on real numbers (the floor 
of a real number x is the largest integer that is 
smaller than or equal to x and the ceiling of 
a real number x is the smallest integer that is 
greater than or equal to x). 



keys 

values 

min()—> 

09:00:00 

Chicago 


09:00:03 

Phoenix 


09j00il3-> 

Houston 

get(09:00:13)- 

'05700:59 

Chicago 


09:01:10 

Houston 

floor(09:05:00) — 

-09:03:13 

Chicago 


09:10:11 

Seattle 

select (7)— >-09:10:25 

Seattle 


09:14:25 

Phoenix 


09:19:32 

Chicago 


09:19:46 

Chicago 

keys(09:15:00, 09:25:00)—► 

09:21:05 

Chicago 


09:22:43 

Seattle 


09:22:54 

Seattle 


09:25:52 

Chicago 

ceiling(09:30:00) —- 

09:35:21 

Chicago 


09:36:14 

Seattle 

max() — 

09:37:44 

Phoenix 


size(09:15:00, 09:25:00) is 5 
rank(09:10:25) is 7 


Examples of ordered symbol-table operations 


Rank and selection. The basic operations for determining where a new key fits in the 
order are the rank operation (find the number of keys less than a given key) and the 
select operation (find the key with a given rank). To test your understanding of their 
meaning, confirm for yourself that both i == rank(select(i )) for all i between 0 
and size()-l and all keys in the table satisfy key == select(rank(key)). We have 
already encountered the need for these operations, in our discussion of sort applica¬ 
tions in section 2.5. For symbol tables, our challenge is to perform these operations 
quickly, intermixed with insertions, deletions, and searches. 
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Range queries. How many keys fall within a given range (between two given keys)? 
Which keys fall in a given range? The two-argument si ze () and keys () methods that 
answer these questions are useful in many applications, particularly in large databases. 
The capability to handle such queries is one prime reason that ordered symbol tables 
are so widely used in practice. 

Exceptional cases. When a method is to return a key and there is no key fitting the de¬ 
scription in the table, our convention is to throw an exception (an alternate approach, 
which is also reasonable, would be to return nul 1 in such cases). For example, mi n (), 
max(), deleteMinO, deleteMaxO, floorO, and ceilingO all throw exceptions if 
the table is empty, as does sel ect(k) if k is less than 0 or not less than si ze(). 

Shorthand methods. As we have already seen with i sEmpty O and contai ns () in our 
basic API, we keep some redundant methods in the API for clarity in client code. For 
economy in the text, we assume that the following default implementations are includ¬ 
ed in any implementation of the ordered symbol-table API unless otherwise specified: 

method default implementation 

void deleteMinO delete(min()) ; 

void deleteMaxO delete(maxO) ; 

int size(Key lo, Key hi) if (hi .compareTo(lo) < 0) 

return 0; 

else if (contains(hi)) 

return rank(hi) - rank(lo) + 1; 
el se 

return rank(hi) - rank(lo); 

Iterable<Key> keys() return keys(min(), max()); 

Default implementations of redundant order-based symbol-table methods 


Key equality (revisited). The best practice in Java is to make compareToO consistent 
with equal s() in all Comparable types. That is, for every pair of values a and b in 
any given Comparabl e type, it should be the case that (a. compareTo(b) == 0) and 
a. equal s (b) have the same value. To avoid any potential ambiguities, we avoid the use 
of equal s() in ordered symbol-table implementations. Instead, we use compareToO 
exclusively to compare keys: we take the boolean expression a. compareTo(b) == 0 to 
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mean “Are a and b equal ?” Typically, such a test marks the successful end of a search for 
a in the symbol table (by finding b). As you saw with sorting algorithms, Java provides 
standard implementations of compareToO for many commonly used types of keys, 
and it is not difficult to develop a compareToO implementation for your own data 
types (see section 2 . 5 ). 

Cost model. Whether we use equal s() (for symbol tables 
where keys are not Comparable) or compareToO (for or¬ 
dered symbol tables with Comparabl e keys), we use the term 
compare to refer to the operation of comparing a symbol- 
table entry against a search key. In most symbol-table imple¬ 
mentations, this operation is in the inner loop. In the few 
cases where that is not the case, we also count array accesses. 

symbol-table implementations are generally character¬ 
ized by their underlying data structures and their implemen¬ 
tations of get() and put(). We do not always provide im¬ 
plementations of all of the other methods in the text because 
many of them make good exercises to test your understanding of the underlying data 
structures. To distinguish implementations, we add a descriptive prefix to ST that refers 
to the implementation in the class name of symbol-table implementations. In clients, 
we use ST to call on a reference implementation unless we wish to refer to a specific 
implementation. You will gradually develop a better feeling for the rationale behind the 
methods in the APIs in the context of the numerous clients and symbol-table imple¬ 
mentations that we present and discuss throughout this chapter and throughout the 
rest of this book. We also discuss alternatives to the various design choices that we have 
described here in the Q&A and exercises. 


Searching cost model. 

When studying symbol-table 
implementations, we count 
compares (equality tests or 
key comparisons). In (rare) 
cases where compares are not 
in the inner loop, we count 
array accesses. 
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Sample clients While we defer detailed consideration of applications to section 
3 . 5 , it is worthwhile to consider some client code before considering implementations. 
Accordingly, we now consider two clients: a test client that we use to trace algorithm 
behavior on small inputs and a performance client that we use to motivate the develop¬ 
ment of efficient implementations. 


public 
{ 


static void main(String[] args) 


ST<String, Integer> st; 

st = new ST<String, Integer>() 


for 
{ 


} 


keys S 
values 0 


Test client. For tracing our algorithms on small inputs we assume that all of our im¬ 
plementations use the test client below, which takes a sequence of strings from standard 
input, builds a symbol table that associates the value i with the i th string in the input, 
and then prints the table. In the traces in the text, we assume that the input is a sequence 

of single-character strings. Most often, we 
use the string "S EARCHEXAMPL E". 
By our conventions, this client associates 
the key S with the value 0, the key R with 
the value 3, and so forth. But E is associated 
with the value 12 (not 1 or 6 ) and A with 
the value 8 (not 2 ) because our associative- 
array convention implies that each key is 
associated with the value used in the most 
recent call to put(). For basic (unordered) 
implementations, the order of the keys in 
the output of this test client is not specified 
(it depends on characteristics of the imple¬ 
mentation); for an ordered symbol table 
the test client prints the keys in sorted order. 
This client is an example of an indexing cli¬ 
ent, a special case of a fundamental symbol- 
table application that we discuss in section 
3-5- 


(int i = 0; ! Stdln . i sEmptyO ; i++) 

String key = Stdln . readStri ng() ; 
st.putCkey, i); 


for (String s : st.keysO) 

StdOut.println(s + " " + st.get(s)); 


Basic symbol-table test client 


M P L E 
9 10 11 12 


output for 
basic symbol table 
(one possibility) 


output for 
ordered 
symbol table 
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Keys, values, and output for test client 
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Performance client. FrequencyCounter (on the next page) is a symbol-table client 
that finds the number of occurrences of each string (having at least as many characters 
as a given threshold length) in a sequence of strings from standard input, then iterates 
through the keys to find the one that occurs the most frequently. This client is an exam¬ 
ple of a dictionary client, an application that we discuss in more detail in section 3 . 5 . 
This client answers a simple question: Which word (no shorter than a given length) oc¬ 
curs most frequently in a given text? Throughout this chapter, we measure performance 
of this client with three reference inputs: the first five lines of C. Dickens’s Tale of Two 
Cities (ti nyTal e. txt), the full text of the book (tal e. txt), and a popular database of 
1 million sentences taken at random from the web that is known as the Leipzig Corpora 
Collection (1 ei pzi glM. txt). For example, here is the content of ti nyTal e. txt: 


% more 
i t was 
i t was 
i t was 
i t was 
i t was 


ti nyTale.txt 

the best of times it was the worst of times 
the age of wisdom it was the age of foolishness 
the epoch of belief it was the epoch of incredulity 
the season of light it was the season of darkness 
the spring of hope it was the winter of despair 


Small test Input 

This text has 60 words taken from 20 distinct words, four of which occur ten times (the 
highest frequency). Given this input, FrequencyCounter will print out any of i t, was, 
the, or of (the one chosen may vary, depending upon characteristics of the symbol- 
table implementation), followed by the frequency, 10 . 

To study performance for the larger inputs, it is clear that two measures are of inter¬ 
est: Each word in the input is used as a search key once, so the total number of words 
in the text is certainly relevant. Second, each distinct word in the input is put into the 



tinyTale.txt 

words distinct 

tale, 

words 

.txt 

distinct 

leipziglM 

words 

.txt 

distinct 

all words 

60 

20 

135,635 

10,679 

21,191,455 

534,580 

at least 8 letters 

3 

3 

14,350 

5,737 

4,239,597 

299,593 

at least 10 letters 

2 

2 

4,582 

2,260 

1,610,829 

165,555 


Characteristics of larger test input streams 
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A symbol-table client 


public class FrequencyCounter 

{ 

public static void main(String[] args) 

{ 

int minlen = Integer.parselnt(args[0]); // key-length cutoff 

ST<String, Integer> st = new ST<String, Integer>(); 
while (!Stdln.isEmptyO) 

{ // Build symbol table and count frequencies. 

String word = Stdln.readString() ; 

if (word.lengthO < minlen) continue; // Ignore short keys. 

if (ist.contains(word)) st.put(word, 1); 

else st.put(word, st.get(word) + 1); 

} 

// Find a key with the highest frequency count. 

String max = 

st.put(max, 0); 

for (String word : st.keysO) 

if (st.get(word) > st.get(max)) 
max = word; 

StdOut.println(max + " " + st.get(max)); 

} 

} 


This ST client counts the frequency of occurrence of the strings in standard input, then prints out 
one that occurs with highest frequency. The command-line argument specifies a lower bound on the 
length of keys considered. 

% java FrequencyCounter 1 < tinyTale.txt 
it 10 

% java FrequencyCounter 8 < tale.txt 
business 122 

% java FrequencyCounter 10 < leipziglM.txt 
government 24763 
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symbol table (and the total number of distinct words in the input gives the size of the 
table after all keys have been inserted), so the total number of distinct words in the 
input stream is certainly relevant. We need to know both of these quantities in order 
to estimate the running time of FrequencyCounter (for a start, see exercise 3 . 1 . 6 ). 
We defer details until we consider some algorithms, but you should have in mind a 
general idea of the needs of typical applications like this one. For example, running 
FrequencyCounter on 1 ei pzi glM. txt for words of length 8 or more involves millions 
of searches in a table with hundreds of thousands of keys and values. A server on the 
web might need to handle billions of transactions on tables with millions of keys and 
values. 

the basic question that this client and these examples raise is the following: Can 
we develop a symbol-table implementation that can handle a huge number of get() 
operations on a large table, which itself was built with a large number of intermixed 
get () and put () operations? If you are only doing a few searches, any implementation 
will do, but you cannot make use of a client like FrequencyCounter for large prob¬ 
lems without a good symbol-table implementation. FrequencyCounter is surrogate 
for a very common situation. Specifically, it has the following characteristics, which are 
shared by many other symbol-table clients: 

■ Search and insert operations are intermixed. 

■ The number of distinct keys is not small. 

■ Substantially more searches than inserts are likely. 

■ Search and insert patterns, though unpredictable, are not random. 

Our goal is to develop symbol-table implementations that make it feasible to use such 
clients to solve typical practical problems. 

Next, we consider two elementary implementations and their performance for 
FrequencyCounter. Then, in the next several sections, you will learn classic imple¬ 
mentations that can achieve excellent performance for such clients, even for huge input 
streams and tables. 
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Sequential search in an unordered linked list One straightforward option 
for the underlying data structure for a symbol table is a linked list of nodes that contain 
keys and values, as in the code on the facing page. To implement g e t ( ), we scan through 
the list, using equal s() to compare the search key with the key in each node in the list. 
If we find the match, we return the associated value; if not, we return nul 1 . To imple¬ 
ment put(), we also scan through the list, using equal s() to compare the client key 
with the key in each node in the list. If we find the match, we update the value associ¬ 
ated with that key to be the value given in the second argument; if not, we create a new 
node with the given key and value and insert it at the beginning of the list. This method 
is known as sequential search: we search by considering the keys in the table one after 
another, using equal s () to test for a match with our search key. 

algorithm 3.1 (Sequent! al SearchST) is an implementation of our basic symbol- 
table API that uses standard list-processing mechanisms, which we have used for el¬ 
ementary data structures in chapter i. We have left the implementations of si ze(), 
keys(), and eager delete() for exercises. You are encouraged to work these exercises 
to cement your understanding of the linked-list data structure and the basic symbol- 
table API. 

Can this linked-list-based implementation handle applications that need large ta¬ 
bles, such as our sample clients? As we have noted, analyzing symbol-table algorithms 
is more complicated than analyzing sorting algorithms because of the difficulty of 
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ALGORITHM 3.1 Sequential search (in an unordered linked list) 


public class SequentialSearchSTcKey, Value> 

{ 

private Node first; // first node in the linked list 

private class Node 

{ // linked-list node 

Key key; 

Value val; 

Node next; 

public Node(Key key, Value val, Node next) 

{ 

this.key = key; 
this.val = val; 
this.next = next; 

} 

} 

public Value get(Key key) 

{ // Search for key, return associated value, 

for (Node x = first; x != null; x = x.next) 
if (key.equals(x.key)) 

return x.val; // search hit 

return null; // search miss 

} 

public void put(Key key, Value val) 

{ // Search for key. Update value if found; grow table if new. 

for (Node x = first; x != null; x = x.next) 

if (key.equals(x.key)) 

{ x.val = val; return; } // Search hit: update val. 

first = new Node(key, val, first); // Search miss: add new node. 

} 

} 


This ST implementation uses a private Node inner class to keep the keys and values in an unordered 
linked list. The get() implementation searches the list sequentially to find whether the key is in the 
table (and returns the associated value if so). The put() implementation also searches the list sequen¬ 
tially to check whether the key is in the table. If so, it updates the associated value; if not, it creates a 
new node with the given key and value and inserts it at the beginning of the list. Implementations of 
si ze(), keys(), and eager del ete() are left for exercises. 
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characterizing the sequence of operations that might be invoked by a given client. As 
noted for FrequencyCounter, the most common situation is that, while search and 
insert patterns are unpredictable, they certainly are not random. For this reason, we pay 
careful attention to worst-case performance. For economy, we use the term search hit to 
refer to a successful search and search miss to refer to an unsuccessful search. 


Proposition A. Search misses and insertions in an (unordered) linked-list sym¬ 
bol table having N key-value pairs both require N compares, and search hits N 
compares in the worst case. In particular, inserting N distinct keys into an initially 
empty linked-list symbol table uses ~N 2 1 2 compares. 

Proof: When searching for a key that is not in the list, we test every key in the table 
against the search key. Because of our policy of disallowing duplicate keys, we need 
to do such a search before each insertion. 


Corollary. Inserting N distinct keys into an initially empty linked-list symbol table 
uses ~N 2 / 2 compares. 


It is true that searches for keys that are in the table need not take linear time. One useful 
measure is to compute the total cost of searching for all of the keys in the table, divided 
by N. This quantity is precisely the expected number of compares required for a search 
under the condition that searches for each key in the table are equally likely. We refer to 
such a search as a random search hit. Though client search patterns are not likely to be 
random, they often are well-described by this model. It is easy to show that the average 
number of compares for a random search hit is ~ N/2: the get() method in algo¬ 
rithm 3.1 uses 1 compare to find the first key, 2 compares to find the second key, and 
so forth, for an average cost of (1 + 2 + ... + N)/ N = (N + l)/2 ~ N/2. 

This analysis strongly indicates that a linked-list implementation with sequential 
search is too slow for it to be used to solve huge problems such as our reference inputs 
with clients like FrequencyCounter. The total number of compares is proportional to 
the product of the number of searches and the number of inserts, which is more than 
10 9 for Tale of Two Cities and more than 10 14 for the Leipzig Corpora. 

As usual, to validate analytic results, we need to run experiments. As an example, 
we study the operation of FrequencyCounter with command-line argument 8 for 
tal e. txt, which involves 14,350 put () operations (recall that every word in the input 
leads to a put(), to update its frequency, and we ignore the cost of easily avoided calls 
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to contai ns()). The symbol table grows to 5,737 keys, so about one-third of the opera¬ 
tions increase the size of the table; the rest are searches. To visualize the performance, we 
use Vi sualAccumulator (see page 95) to plot two points corresponding to each put() 
operation as follows: for the / th put() operation we plot a gray point with x coordinate 
i and y coordinate the number of key compares it uses and a red point with x coordi¬ 
nate i and y coordinate the cumulative average number of key compares used for the 
first i put() operations. As with any scientific data, there is a great deal of information 
in this data for us to investigate (this plot has 14,350 gray points and 14,350 red points). 
In this context, our primary interest is that the plot validates our hypothesis that about 
half the list is accessed for the average put() operation. The actual total is slightly lower 
than half, but this fact (and the precise shape of the curves) is perhaps best explained by 
characteristics of the application, not our algorithms (see exercise 3.1.36). 

While detailed characterization of performance for particular clients can be com¬ 
plicated, specific hypotheses are easy to formulate and to test for FrequencyCount 
with our reference inputs or with randomly ordered inputs, using a client like the 
DoublingTest client that we introduced in chapter 1. We will reserve such testing 
for exercises and for the more sophisticated implementations that follow. If you are not 
already convinced that we need faster implementations, be sure to work these exercises 
(or just run FrequencyCounter with SequentialSearchST on 1 eipziglM.txt!). 
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Costsforjava FrequencyCounter 8 < tale.txt using Sequenti al SearchST 
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Binary search in an ordered array Next, we consider a full implementation 
of our ordered symbol-table API. The underlying data structure is a pair of parallel 
arrays, one for the keys and one for the values, algorithm 3.2 (Bi narySearchST) on 
the facing page keeps Comparabl e keys in order in the array, then uses array indexing 
to enable fast implementation of get() and other operations. 

The heart of the implementation is the rank() method, which returns the number 
of keys smaller than a given key. For get(), the rank tells us precisely where the key is 
to be found if it is in the table (and, if it is not there, that it is not in the table). 

For put (), the rank tells us precisely where to update the value when the key is in the 
table, and precisely where to put the key when the key is not in the table. We move all 
larger keys over one position to make room (working from back to front) and insert the 
given key and value into the proper positions in their respective arrays. Again, studying 
Bi narySearchST in conjunction with a trace of our test client is an instructive intro¬ 
duction to this data structure. 

This code maintains parallel arrays of keys and values (see exercise 3 . 1.12 for an 
alternative). As with our implementations of generic stacks and queues in chapter 1 , 
this code carries the inconvenience of having to create a Key array of type Comparabl e 
and a Val ue array of type Object, and to cast them back to Key [] and Val ue [] in the 
constructor. As usual, we can use array resizing so that clients do not have to be con¬ 
cerned with the size of the array (noting, as you shall see, that this method is too slow 
to use with large arrays). 
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ALGORITHM 3.2 Binary search (in an ordered array) 


public class BinarySearchST<Key extends Comparable<Key>, Value> 

{ 

private Key[] keys; 
private Value[] vals; 
private int N; 

public BinarySearchSTCint capacity) 

{ // See Algorithm 1.1 for standard array-resizing code, 

keys = (Key[]) new Comparable[capacity]; 
vals = (Value[]) new Object[capacity]; 

} 

public int sizeO 
{ return N; } 

public Value get(Key key) 

{ 

if (isEmptyO) return null; 
int i = rank(key); 

if (i < N && keys[i].compareTo(key) == 0) return vals[i]; 
else return nul1 ; 

} 

public int rank(Key key) 

// See page 381. 

public void put(Key key, Value val) 

{ // Search for key. Update value if found; grow table if new. 

int i = rank(key); 

if (i < N && keys[i].compareTo(key) == 0) 

{ vals[i] = val; return; } 
for (int j = N; j > i ; j—) 

{ keys[j] = keys[j-l]; vals[j] = vals[j-l]; } 
keys[i] = key; vals[i] = val; 

N++; 

} 

public void delete(Key key) 

// See Exercise 3.1.16 for this method. 


This ST implementation keeps the keys and values in parallel arrays. The put() implementation 
moves larger keys one position to the right before growing the table as in the array-based stack imple¬ 
mentation in section 1.3. Array-resizing code is omitted here. 


www.it-ebooks.info 






380 


CHAPTER 3 ■ Searching 


Binary search. The reason that we keep keys in an ordered array is so that we can 
use array indexing to dramatically reduce the number of compares required for each 
search, using the classic and venerable binary search algorithm that we used as an exam¬ 
ple in chapter 1 . We maintain indices into 
the sorted key array that delimit the subar¬ 
ray that might contain the search key. To 
search, we compare the search key against 
the key in the middle of the subarray. If the 
search key is less than the key in the middle, 
we search in the left half of the subarray; if 
the search key is greater than the key in the 
middle, we search in the right half of the 
subarray; otherwise the key in the middle is 
equal to the search key. The rank() code on 
the facing page uses binary search to com¬ 
plete the symbol-table implementation just discussed. This implementation is worthy 
of careful study. To study it, we start with the equivalent recursive code at left. A call to 
rank(key , 0, N-l) does the same sequence of compares as a call to the nonrecursive 
implementation in algorithm 3 . 2 , but this alternate version better exposes the struc¬ 
ture of the algorithm, as discussed in section 1.1. This recursive rank() preserves the 
following properties: 

■ If key is in the table, rank() returns its index in the table, which is the same as 
the number of keys in the table that are smaller than key. 

■ If key is not in the table, rank() also returns the number of keys in the table 
that are smaller than key. 

Taking the time to convince yourself that the nonrecursive rank() in algorithm 3.2 
operates as expected (either by proving that it is equivalent to the recursive version or 
by proving directly that the loop always terminates with the value of 1 o precisely equal 
to the number of keys in the table that are smaller than key) is a worthwhile exercise for 
any programmer. {Hint: Note that 1 o starts at 0 and never decreases.) 

Other operations. Since the keys are kept in an ordered array, most of the order-based 
operations are compact and straightforward, as you can see from the code on page 382. For 
example, a call to select(k) just returns keys [k] .We have left del ete() and floor 0 
as exercises. You are encouraged to study the implementation of cei lingO and the 
two-argument keys () and to work these exercises to cement your understanding of the 
ordered symbol-table API and this implementation. 


public int rank(Key key, int lo, int hi) 
{ 

if (hi < lo) return lo; 
int mid = lo + (hi - lo) / 2; 
int cmp = key.compareTo(keys[mid]); 
if (cmp < 0) 

return rank(key, lo, mid-1); 
else if (cmp > 0) 

return rank(key, mid+1, hi); 
else return mid; 


Recursive binary search 
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ALGORITHM 3.2 (continued) Binary search in an ordered array (iterative) 


public int rank(Key key) 

{ 

int lo = 0, hi = N—1; 
while (lo <= hi) 

{ 

int mid = lo + (hi - lo) / 2; 
int cmp = key.compareTo(keys[mid]); 
if (cmp < 0) hi = mid - 1; 

else if (cmp >0) lo = mid + 1; 
else return mid; 

} 

return lo; 


This method uses the classic method described in the text to compute the number of keys in the table 
that are smaller than key. Compare key with the key in the middle: if it is equal, return its index; if it 
is less, look in the left half; if it is greater, look in the right half. 


successful search for P 0 12 

lo hi mid 

0 9 4 ACE 

5 9 7 

5 6 5 

6 6 6 


unsuccessful search for Q 

lo hi mid 

0 9 4 ACE 

5 9 7 

5 6 5 

7 6 6 


X 


loop exits with 1 o 


keys [] 


5 4 5 6 7 8 9 



entries in black 

are a[lo. . hi ] 


^ entry in red is a [mi d] 
loop exits with keys [mi d] = P: return 


H L M P R S X 

M P R S X 

M P 
M P 

> hi: return 7 


Trace of binary search for rank in an ordered array 
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ALGORITHM 3.2 (continued) Ordered symbol-table operations for binary search 


public Key min() 

{ return keys[0]; } 

public Key max() 

{ return keys[N-l]; } 

public Key select(int k) 

{ return keys[k]; } 

public Key ceiling(Key key) 

{ 

int i = rank(key); 
return keys[i]; 

} 

public Key floor(Key key) 

// See Exercise 3.1.17. 

public Key delete(Key key) 

// See Exercise 3.1.16. 

public Iterable<Key> keys(Key lo, Key hi) 

{ 

Queue<Key> q = new Queue<Key>(); 
for (int i = rank(lo); i < rank(hi); i++) 
q.enqueue(keys[i]); 
if (contains(hi)) 

q.enqueue(keys[rank(hi)]); 
return q; 

} 


These methods, along with the methods of exercise 3 . 1.16 and exercise 3 . 1 . 17 , complete the imple¬ 
mentation of our (ordered) symbol-table API using binary search in an ordered array. The mi n(), 
max(), and select () methods are trivial, just amounting to returning the appropriate key from its 
known position in the array. The rank() method, which is the basis of binary search, plays a central 
role in the others. The floor() and deleteO implementations, left for exercises, are more compli¬ 
cated, but still straightforward. 
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Analysis of binary search The recursive implementation of rank() also leads 
to an immediate argument that binary search guarantees fast search, because it cor¬ 
responds to a recurrence relation that describes an upper bound on the number of 
compares. 

Proposition B. Binary search in an ordered array with N keys uses no more than 
IgN + 1 compares for a search (successful or unsuccessful). 

Proof: This analysis is similar to (but simpler than) the analysis of mergesort 
(proposition f in chapter 2). Let C(N) be the number of compares needed to 
search for a key in a symbol table of size N. We have C(0) = 0, C(l) = 1, and for N > 

0 we can write a recurrence relationship that directly mirrors the recursive method: 

C(N) < C([N/2\) + 1 

Whether the search goes to the left or to the right, the size of the subarray is no 
more than [iV/2j, and we use one compare to check for equality and to choose 
whether to go left or right. When N is one less than a power of 2 (say N = 2"— 1), 
this recurrence is not difficult to solve. First, since [N/2\ = 2" _1 — 1, we have 

C(2" —1) < C(2" -1 —1) + 1 

Applying the same equation to the first term on the right, we have 

C(2" — 1) < C(2”“ 2 — 1) + 1 + 1 
Repeating the previous step 11 — 2 additional times gives 

C(2" — 1) < C(2°) + n 
which leaves us with the solution 

C(N) = C( 2 ") < n + 1 < lgN+ 1 

Exact solutions for general N are more complicated, but it is not difficult to extend 
this argument to establish the stated property for all values of N (see exercise 
3.1.20). With binary search, we achieve a logarithmic-time search guarantee. 


The implementation just given for cei 1 i ng() is based on a single call to rank(), and 
the default two-argument si ze() implementation calls rank() twice, so this proof also 
establishes that these operations (and floorO) are supported in guaranteed logarith¬ 
mic time (mi n(), maxQ, and sel ect() are constant-time operations). 
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method 

order of gro 
of running t 

put() 

N 

get() 

log N 

deleteO 

N 

containsO 

log N 

size() 

1 

mi n() 

1 

max() 

1 

floor() 

log N 

cei 1 i ng() 

log N 

rank() 

log N 

selectO 

1 

deleteMin() 

N 

deleteMax() 

1 


Despite its guaranteed logarithmic search, Bi narySearchST 
still does not enable us to use clients like FrequencyCounter to 
solve huge problems, because the put() method is too slow. Bi¬ 
nary search reduces the number of compares, but not the running 
time, because its use does not change the fact that the number of 
array accesses required to build a symbol table in an ordered ar¬ 
ray is quadratic in the size of the array when keys are randomly 
ordered (and in typical practical situations where the keys, while 
not random, are well-described by this model). 

Proposition B (continued). Inserting a new key into an or¬ 
dered array of size N uses ~ 2 N array accesses in the worst 
case, so inserting N keys into an initially empty table uses ~ 
N 2 array accesses in the worst case. 

Proof: Same as for proposition a. 


For Tale of Two Cities, with over 10 4 distinct keys, the cost to build 
the table is nearly 10 s array accesses, and for the Leipzig project, 
with over 10 6 distinct keys, the cost to build the table is over 10 11 
array accesses. While not quite prohibitive on modern computers, these costs are ex¬ 
tremely (and unnecessarily) high. 

Returning to the cost of the put() operations for FrequencyCounter for words of 
length 8 or more, we see a reduction in the average cost from 2,246 compares (plus 
array accesses) per operation for Sequent! al SearchST to 484 for Bi narySearchST. 
As before, this cost is even better than would be predicted by analysis, and the extra 
improvement is likely again explained by properties of the application (see exercise 
3 . 1 . 36 ). This improvement is impressive, but we can do much better, as you shall see. 


BinarySearchST costs 


5737 —1 





operations 

Costsforjava FrequencyCounter 8 < tale.txt using BinarySearchST 


— -*—484 

14350 
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Preview Binary search is typically far better than sequential search and is the meth¬ 
od of choice in numerous practical applications. For a static table (with no insert op¬ 
erations allowed), it is worthwhile to initialize and sort the table, as in the version of 
binary search that we considered in chapter i (see page 99). Even when the bulk of 
the key-value pairs is known before the bulk of the searches (a common situation in 
applications), it is worthwhile to add to Bi narySearchST a constructor that initial¬ 
izes and sorts the table (see exercise 3.1.12). Still, binary search is infeasible for use 
in many other applications. For example, it fails for our Leipzig Corpora application 
because searches and inserts are intermixed and the table size is too large. As we have 
emphasized, typical modern search clients require symbol tables that can support fast 
implementations of both search and insert. That is, we need to be able to build huge 
tables where we can insert (and perhaps remove) key-value pairs in unpredictable pat¬ 
terns, intermixed with searches. 

The table below summarizes performance characteristics for the elementary sym¬ 
bol-table implementations considered in this section. The table entries give the leading 
term of the cost (number of array accesses for binary search, number of compares for 
the others), which implies the order of growth of the running time. 


algorithm 
(data structure) 


worst-case cost 
(after N inserts) 

search insert 


average-case cost 
(after N random inserts) 

search hit insert 


efficiently 
support ordered 
operations? 


sequential search 
(unordered linked list) 

N 

N 

N/2 

N 

no 

binary search 
(ordered array) 

lg N 

2 N 

lg N 

N 

yes 


Cost summary for basic symbol-table implementations 


The central question is whether we can devise algorithms and data structures that 
achieve logarithmic performance for both search and insert. The answer is a resound¬ 
ing yes! Providing that answer is the main thrust of this chapter. Along with the fast sort 
capability discussed in chapter 2, fast symbol-table search/insert is one of the most 
important contributions of algorithmics and one of the most important steps toward 
the development of the rich computational infrastructure that we now enjoy. 

How can we achieve this goal? To support efficient insertion, it seems that we need a 
linked structure. But a singly linked list forecloses the use of binary search, because the 
efficiency of binary search depends on our ability to get to the middle of any subarray 


www.it-ebooks.info 



386 


CHAPTER 3 ■ Searching 


quickly via indexing (and the only way to get to the middle of a singly linked list is to 
follow links). To combine the efficiency of binary search with the flexibility of linked 
structures, we need more complicated data structures. That combination is provided 
both by binary search trees, the subject of the next two sections, and by hash tables, the 
subject of section 3 . 4 . 

We consider six symbol-table implementations in this chapter, so a brief preview is 
in order. The table below is a list of the data structures, along with the primary reasons 
in favor of and against using each for an application. They appear in the order in which 
we consider them. 

We will get into more detail on properties of the algorithms and implementations 
as we discuss them, but the brief characterizations in this table will help you keep them 
in a broader context as you learn them. The bottom line is that we have several fast 
symbol-table implementations that can be and are used to great effect in countless 
applications. 


underlying 


data structure 

implementation 

pros 

cons 

linked list 
(sequential 

Sequent!alSearchST 

best for tiny STs 

slow for large STs 

search) 




ordered array 
(binary search) 

BinarySearchST 

optimal search 
and space, 
order-based ops 

slow insert 

binary 
search tree 

BST 

easy to 
implement, 
order-based ops 

no guarantees 
space for links 

balanced 

BST 

RedBlackBST 

optimal search 
and insert, 
order-based ops 

space for links 

hash table 

SeparateChainf ngHashST 
LinearProbingHashST 

fast search/insert 
for common types 
of data 

need hash for each type 
no order-based ops 
space for links/empty 


Pros and cons of symbol-table implementations 
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Q&A 


Q. Why not use an Item type that implements Comparable for symbol tables, in the 
same way as we did for priority queues in section 2 . 4 , instead of having separate keys 
and values ? 

A. That is also a reasonable option. These two approaches illustrate two different ways 
to associate information with keys—we can do so implicitly by building a data type that 
includes the key or explicitly by separating keys from values. For symbol tables, we have 
chosen to highlight the associative array abstraction. Note also that a client specifies just 
a key in search, not a key-value aggregation. 

Q. Why bother with equal s() ? Why not just use compareToO throughout? 

A. Not all data types lead to key values that are easy to compare, even though having a 
symbol table still might make sense. To take an extreme example, you may wish to use 
pictures or songs as keys. There is no natural way to compare them, but we can certainly 
test equality (with some work). 

Q. Why not allow keys to take the value null? 

A. We assume that Key is an Object because we use it to invoke compareToO or 
equal s CD - But a call like a. compareTo(b) would cause a null pointer exception if a is 
null . By ruling out this possibility, we allow for simpler client code. 

Q. Why not use a method like the 1 ess 0 method that we used for sorting? 

A. Equality plays a special role in symbol tables, so we also would need a method for 
testing equality. To avoid proliferation of methods that have essentially the same func¬ 
tion, we adopt the built-in Java methods equalsO and compareToO. 

Q. Why not declare key [] asObject[] (instead of Comparabl e []) in Bi narySearchST 
before casting, in the same way that val [] is declared as Object? 

A. Good question. If you do so, you will get a Cl assCastExcepti on because keys need 
to be Comparable (to ensure that entries in key [] have a compareToO method). Thus, 
declaring key[] as Comparabl e[] is required. Delving into the details of program¬ 
ming-language design to explain the reasons would take us somewhat off topic. We use 
precisely this idiom (and nothing more complicated) in any code that uses Comparabl e 
generics and arrays throughout this book. 
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Q&A (continued) 


Q. What if we need to associate multiple values with the same key? For example, if we 
use Date as a key in an application, wouldn’t we have to process equal keys? 

A. Maybe, maybe not. For example, you can’t have two trains arrive at the station on 
the same track at the same time (but they could arrive on different tracks at the same 
time). There are two ways to handle the situation: use some other information to dis¬ 
ambiguate or make the value a Queue of values having the same key. We consider ap¬ 
plications in detail in section 3.5. 

Q. Presorting the table as discussed on page 385 seems like a good idea. Why relegate 
that to an exercise (see exercise 3.1.12)? 

A. Indeed, this may be the method of choice in some applications. But adding a slow 
insert method to a data structure designed for fast search “for convenience” is a per¬ 
formance trap, because an unsuspecting client might intermix searches and inserts in 
a huge table and experience quadratic performance. Such traps are all too common, 
so that “buyer beware” is certainly appropriate when using software developed by oth¬ 
ers, particularly when interfaces are too wide. This problem becomes acute when a 
large number of methods are included “for convenience” leaving performance traps 
throughout, while a client might expect efficient implementations of all methods. Java’s 
ArrayLi st class is an example (see exercise 3.5.27). 
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EXERCISES 


3 . 1.1 Write a client that creates a symbol table mapping letter grades to numerical 
scores, as in the table below, then reads from standard input a list of letter grades and 
computes and prints the GPA (the average of the numbers corresponding to the grades). 


A+ 

A 

A- 

B+ 

B 

B- 

C+ 

C 

C- 

D 

F 

4.33 

4.00 

3.67 

3.33 

3.00 

2.67 

2.33 

2.00 

1.67 

1.00 

0.00 


3 . 1.2 Develop a symbol-table implementation Ar rayST that uses an (unordered) array 
as the underlying data structure to implement our basic symbol-table API. 


3 . 1.3 Develop a symbol-table implementation OrderedSequential SearchST that 
uses an ordered linked list as the underlying data structure to implement our ordered 
symbol-table API. 

3 . 1.4 Develop Time and Event ADTs that allow processing of data as in the example 
illustrated on page 367 . 

3 . 1.5 Implement size(), delete(), andkeys()for SequentialSearchST. 

3 . 1.6 Give the number of calls to put() and get() issued by FrequencyCounter, as a 
function of the number W of words and the number D of distinct words in the input. 

3 . 1.7 What is the average number of distinct keys that FrequencyCounter will find 
among N random nonnegative integers less than 1,000, for N=10, 10 2 , 10 3 , 10 4 , 10 5 , and 
10 6 ? 

3 . 1.8 What is the most frequently used word of ten letters or more in Tale of Two Cities ? 

3 . 1.9 Add code to FrequencyCounter to keep track of the last call to put(). Print the 
last word inserted and the number of words that were processed in the input stream 
prior to this insertion. Run your program for tal e . txt with length cutoffs 1, 8 , and 10. 

3 . 1.10 Give a trace of the process of inserting the keys E ASYQUESTI0N into an 
initially empty table using Sequent! al SearchST. How many compares are involved? 

3 . 1.11 Give a trace of the process of inserting the keys E ASYQUESTI0N into 
an initially empty table using Bi narySearchST. How many compares are involved? 

3 . 1.12 Modify Bi narySearchST to maintain one array of Item objects that contain 
keys and values, rather than two parallel arrays. Add a constructor that takes an array of 
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EXERCISES (continued) 


Item values as argument and uses mergesort to sort the array. 

3.1.13 Which of the symbol-table implementations in this section would you use for 
an application that does 10 3 put() operations and 10 6 get() operations, randomly 
intermixed? Justify your answer. 

3.1.14 Which of the symbol-table implementations in this section would you use for 
an application that does 10 6 put() operations and 10 3 get() operations, randomly 
intermixed? Justify your answer. 

3.1.15 Assume that searches are 1,000 times more frequent than insertions for a 
BinarySearchST client. Estimate the percentage of the total time that is devoted to 
insertions, when the number of searches is 10 3 , 10 6 , and 10 9 . 

3.1.16 Implement the del ete() method for Bi narySearchST. 

3.1.17 Implement the floorf) method for Bi narySearchST. 

3.1.18 Prove that the rankO method in Bi narySearchST is correct. 

3.1.19 Modify FrequencyCounter to print all of the values having the highest fre¬ 
quency of occurrence, not just one of them. Hint: Use a Queue. 

3.1.20 Complete the proof of proposition b (show that it holds for all values of N). 
Hint: Start by showing that C{N) is monotonic: C(N) < C(NT1) for all N > 0. 
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CREATIVE PROBLEMS 


3.1.21 Memory usage. Compare the memory usage of Bi narySearchST with that of 
Sequent! al SearchST for N key-value pairs, under the assumptions described in sec¬ 
tion 1 . 4 . Do not count the memory for the keys and values themselves, but do count 
references to them. For Bi narySearchST, assume that array resizing is used, so that the 
array is between 25 percent and 100 percent full. 

3.1.22 Self-organizing search. A self-organizing search algorithm is one that rearrang¬ 
es items to make those that are accessed frequently likely to be found early in the search. 
Modify your search implementation for exercise 3 . 1.2 to perform the following action 
on every search hit: move the key-value pair found to the beginning of the list, moving 
all pairs between the beginning of the list and the vacated position to the right one posi¬ 
tion. This procedure is called the move-to-front heuristic. 

3.1.23 Analysis of binary search. Prove that the maximum number of compares used 
for a binary search in a table of size N is precisely the number of bits in the binary rep¬ 
resentation of N, because the operation of shifting 1 bit to the right converts the binary 
representation of N into the binary representation of |N/2j. 

3.1.24 Interpolation search. Suppose that arithmetic operations are allowed on keys 
(for example, they may be Doubl e or Integer values). Write a version of binary search 
that mimics the process of looking near the beginning of a dictionary when the word 
begins with a letter near the beginning of the alphabet. Specifically, if k x is the key value 
sought, k lo is the key value of the first key in the table, and k hi is the key value of the 
last key in the table, look first [( k x — k lo )/(k hl — k h ) J-way through the table, not half¬ 
way. Test your implementation against Bi narySearchST for FrequencyCounter using 
SearchCompare. 

3.1.25 Software caching. Since the default implementation of contai ns 0 calls get (), 
the inner loop of FrequencyCounter 

if (!st.contains(word)) st.putfword, 1); 

else st.putfword, st.get(word) + 1); 

leads to two or three searches for the same key. To enable clear client code like this 
without sacrificing efficiency, we can use a technique known as software caching, where 
we save the location of the most recently accessed key in an instance variable. Modify 
Sequenti al SearchST and Bi narySearchST to take advantage of this idea. 
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CREATIVE PROBLEMS (continued) 


3.1.26 Frequency count from a dictionary. Modify FrequencyCounter to take the 
name of a dictionary file as its argument, count frequencies of the words from standard 
input that are also in that file, and print two tables of the words with their frequencies, 
one sorted by frequency, the other sorted in the order found in the dictionary file. 

3.1.27 Small tables. Suppose that a Bi narySearchST client has S search operations 
and N distinct keys. Give the order of growth of S such that the cost of building the table 
is the same as the cost of all the searches. 

3.1.28 Ordered insertions. Modify Bi narySearchST so that inserting a key that is larg¬ 
er than all keys in the table takes constant time (so that building a table by calling put 0 
for keys that are in order takes linear time). 

3.1.29 Test client. Write a test client TestBinarySearch.java for use in test¬ 
ing the implementations of min(), max(), floor(), ceilingO, selectO, rank(), 
del eteMi n(), del eteMaxO, and keys() that are given in the text. Start with the stan¬ 
dard indexing client given on page 370. Add code to take additional command-line argu¬ 
ments, as appropriate. 

3.1.30 Certification. Add assert statements to Bi narySearchST to check algorithm 
invariants and data structure integrity after every insertion and deletion. For example, 
every indexi should always be equal to rankfsel ect(i )) and the array should always 
be in order. 
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EXPERIMENTS 


3.1.31 Performance driver. Write a performance driver program that uses put() to 
fill a symbol table, then uses get() such that each key in the table is hit an average of 
ten times and there is about the same number of misses, doing so multiple times on 
random sequences of string keys of various lengths ranging from 2 to 50 characters; 
measures the time taken for each run; and prints out or plots the average running times. 

3.1.32 Exercise driver. Write an exercise driver program that uses the methods in our 
ordered symbol-table API on difficult or pathological cases that might turn up in prac¬ 
tical applications. Simple examples include key sequences that are already in order, key 
sequences in reverse order, key sequences where all keys are the same, and keys consist¬ 
ing of only two distinct values. 

3.1.33 Driver for self-organizing search. Write a driver program for self-organizing 
search implementations (see exercise 3 . 1 . 22 ) that uses get() to fill a symbol table 
with /V keys, then does 10 N successful searches according to a predefined probability 
distribution. Use this driver to compare the running time of your implementation from 
exercise 3 . 1.22 with Bi narySearchST for N = 10 3 , 10 4 , 10 5 , and 10 6 using the prob¬ 
ability distribution where search hits the i th smallest key with probability 1/2'. 

3.1.34 Zipf’s law. Do the previous exercise for the probability distribution where 
search hits the z’th smallest key with probability 1/(;H V ) where H v is a Harmonic num¬ 
ber (see page 185). This distribution is called Zipf’s law. Compare the move-to-ff ont heu¬ 
ristic with the optimal arrangement for the distributions in the previous exercise, which 
is to keep the keys in increasing order (decreasing order of their expected frequency). 

3.1.35 Performance validation I. Run doubling tests that use the first N words of Tale 
of Two Cities for various values of N to test the hypothesis that the running time of 
FrequencyCounter is quadratic when it uses Sequent!alSearchST for its symbol 
table. 

3.1.36 Performance validation II. Explain why the performance of Bi narySearchST 
and Sequent!alSearchST for FrequencyCounter is even better than predicted by 
analysis. 

3.1.37 Put/get ratio. Determine empirically the ratio of the amount of time that 
Bi narySearchST spends on put() operations to the time that it spends on get() op¬ 
erations when FrequencyCounter is used to find the frequency of occurrence of values 
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EXPERIMENTS (continued) 


in 1 million random M-bit i nt values, for M = 10,20, and 30. Answer the same question 
for tal e. txt and compare the results. 

3.1.38 Amortized cost plots. Develop instrumentation for FrequencyCounter, 
Sequenti al SearchST, and Bi narySearchST so that you can produce plots like the 
ones in this section showing the cost of each put() operation during the computation. 

3.1.39 Actual timings. Instrument FrequencyCounter to use Stopwatch and StdDraw 
to make a plot where the x-axis is the number of calls on get () or put () and the y-axis 
is the total running time, with a point plotted of the cumulative time after each call. 
Run your program for Tale of Two Cities using Sequenti al SearchST and again using 
Bi narySearchST and discuss the results. Note: Sharp jumps in the curve may be ex¬ 
plained by caching, which is beyond the scope of this question. 

3.1.40 Crossover to binary search. Find the values of N for which binary search in a 
symbol table of size N becomes 10, 100, and 1,000 times faster than sequential search. 
Predict the values with analysis and verify them experimentally. 

3.1.41 Crossover to interpolation search. Find the values of N for which interpolation 
search in a symbol table of size N becomes 1,2, and 10 times faster than binary search, 
assuming the keys to be random 32-bit integers (see exercise 3 . 1 . 24 ). Predict the values 
with analysis, and verify them experimentally. 
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3.2 BINARY SEARCH TREES 



in this section, we will examine a symbol-table implementation that combines the 
flexibility of insertion in a linked list with the efficiency of search in an ordered array 
Specifically, using two links per node (instead of the one link per node found in linked 
lists) leads to an efficient symbol-table implementation based on the binary search tree 
data structure, which qualifies as one of the most fundamental al¬ 
gorithms in computer science. 

To begin, we define basic terminology. We are working with data 
structures made up of nodes that contain links that are either null 
or references to other nodes. In a binary tree, we have the restric¬ 
tion that every node is pointed to by just one other node, which is 
called its parent (except for one node, the root, which has no nodes 
pointing to it), and that each node has exactly two links, which are 
called its left and right links, that point to nodes called its left child 
and right child, respectively. Although links point to nodes, we can 
view each link as pointing to a binary tree, the tree whose root is the referenced node. 
Thus, we can define a binary tree as a either a null link or a node with a left link and 
a right link, each references to (disjoint) subtrees that are themselves binary trees. In a 
binary search tree, each node also has a key and a value, with an ordering restriction to 
support efficient search. 

Definition. A binary search tree (BST) is a binary tree where each node has a 
Comparabl e key (and an associated value) and satisfies the restriction that the key 
in any node is larger than the keys in all nodes in that node’s left subtree and small¬ 
er than the keys in all nodes in that node’s right subtree. 



Anatomy of a binary tree 


parent of A and R 
left link \_ 

of E 



key 


R)9- 


X 


\ value 
associated 
with R 


/ \ 

keys smaller than E keys larger than E 


Anatomy of a binary search tree 


We draw BSTs with keys in the nodes and use terminology 
such as “A is the left child of E” that associates nodes with keys. 
Lines connecting the nodes represent links, and we give the 
value associated with a key in black, beside the nodes (sup¬ 
pressing the value as dictated by context). Each node’s links 
connect it to nodes below it on the page, except for null links, 
which are short segments at the bottom. As usual, our exam¬ 
ples use the single-letter keys that are generated by our index¬ 
ing test client. 
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Basic implementation algorithm 3.3 defines the BST data structure that we 
use throughout this section to implement the ordered symbol-table API. We begin by 
considering this classic data structure definition and the characteristic associated im¬ 
plementations of the getO (search) and put() (insert) methods. 

Representation. We define a private nested class to define nodes in BSTs, just as we 
did for linked lists. Each node contains a key, a value, a left link, a right link, and a node 
count (when relevant, we include node counts in red above the node 
in our drawings). The left link points to a BST for items with smaller 
keys, and the right link points to a BST for items with larger keys. 

The instance variable N gives the node count in the subtree rooted at 
the node. This field facilitates the implementation of various ordered 
symbol-table operations, as you will see. The private si ze() method 
in algorithm 3.3 is implemented to assign the value 0 to null links, 
so that we can maintain this field by making sure that the invariant 

size(x) = size(x.left) + size(x.right) + 1 

holds for every node x in the tree. 

A BST represents a set of keys (and associated values), and there 
are many different BSTs that represent the same set. If we project the 
keys in a BST such that all keys in each node’s left subtree appear to 
the left of the key in the node and all keys in each node’s right subtree 
appear to the right of the key in the node, then we always get the keys 
in sorted order. We take advantage of the flexibility inherent in having 
many BSTs represent this sorted order to develop efficient algorithms 
for building and using BSTs. 

Search. As usual, when we search for a key in a symbol table, we have one of two 
possible outcomes. If a node containing the key is in the table, we have a search hit, so 
we return the associated vakie. Otherwise, we have a search miss (and return nul 1). A 
recursive algorithm to search for a key in a BST follows immediately from the recursive 
structure: if the tree is empty, we have a search miss; if the search key is equal to the key 
at the root, we have a search hit. Otherwise, we search (recursively) in the appropriate 
subtree, moving left if the search key is smaller, right if it is larger. The recursive get () 
method on page 399 implements this algorithm directly. It takes a node (root of a subtree) 
as first argument and a key as second argument, starting with the root of the tree and 
the search key. The code maintains the invariant that no parts of the tree other than the 
subtree rooted at the current node can have a node whose key is equal to the search key. 
just as the size of the interval in binary search shrinks by about half on each iteration, 


node count N 8 



8 



Two BSTs that represent 
the same set of keys 
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ALGORITHM 3.3 Binary search tree symbol table 


public class BST<Key extends Comparable<Key>, Value> 

{ 

private Node root; // root of BST 

private class Node 

{ 

private Key key; // key 

private Value val; // associated value 

private Node left, right; // links to subtrees 

private int N; // # nodes in subtree rooted here 

public Node(Key key, Value val, int N) 

{ this.key = key; this.val = val; this.N = N; } 

} 

public int size() 

{ return size(root); } 

private int size(Node x) 

{ 

if (x == null) return 0; 
else return x.N; 

} 

public Value get(Key key) 

// See page 399. 

public void putfKey key, Value val) 

// See page 399. 

// See page 407 for min(), max(), floor(), and ceilingC). 

// See page 409 for selectO and rank(). 

// See page 411 for deleteO, deleteMinO, and deleteMax(). 

// See page 413 for keys(). 

} 


This implementation of the ordered symbol-table API uses a binary search tree built from Node ob¬ 
jects that each contain a key, associated value, two links, and a node count N. Each Node is the root 
of a subtree containing N nodes, with its left link pointing to a Node that is the root of a subtree with 
smaller keys and its right link pointing to a Node that is the root of a subtree with larger keys. The 
instance variable root points to the Node at the root of the BST (which has all the keys and associ¬ 
ated values in the symbol table). Implementations of other methods appear throughout this section. 
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ALGORITHM 3.3 (continued) Search and insert for BSTs 


public Value get(Key key) 

{ return getfroot, key); } 

private Value get(Node x, Key key) 

{ // Return value associated with key in the subtree rooted at x; 

// return null if key not present in subtree rooted at x. 

if (x == null) return null; 
int cmp = key.compareTo(x.key); 
if (cmp < 0) return get(x.left, key); 

else if (cmp > 0) return get(x.right, key); 
else return x.val; 

} 

public void put(Key key, Value val) 

{ // Search for key. Update value if found; grow table if new. 

root = putfroot, key, val); 

} 

private Node put(Node x, Key key, Value val) 

{ 

// Change key’s value to val if key in subtree rooted at x. 

// Otherwise, add new node to subtree associating key with val. 
if (x == null) return new Node(key, val, 1); 
int cmp = key.compareTo(x.key); 

if (cmp < 0) x.left = putfx.left, key, val); 

else if (cmp > 0) x.right = put(x.right, key, val); 
else x.val = val; 

x.N = size(x.left) + size(x.right) + 1; 
return x; 

} 


These implementations of get() and put() for the symbol-table API are characteristic recursive 
BST methods that also serve as models for several other implementations that we consider later in 
the chapter. Each method can be understood as both working code and a proof by induction of the 
inductive hypothesis in the opening comment. 
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successful search for R 



unsuccessful search for T 



so T is not in tree 
(search miss) 


Search hit (left) and search miss (right) in a BST 

the size of the subtree rooted at the current node when searching in a BST shrinks when 
we go down the tree (by about half, ideally, but at least by one). The procedure stops 
either when a node containing the search key is found (search hit) or when the current 
subtree becomes empty (search miss). Starting at the top, the search procedure at each 
node involves a recursive invocation for one of that node’s children, so the search de¬ 
fines a path through the tree. For a search hit, the path terminates at the node contain¬ 
ing the key. For a search miss, the path terminates at a null link. 

Insert. The search code in algorithm 3.3 is almost as simple as binary search; that 
simplicity is an essential feature of BSTs. A more important essential feature of BSTs is 
that insert is not much more difficult to implement than search. Indeed, a search for a 
key not in the tree ends at a null link, and all that we need to do is replace that link with 
a new node containing the key (see the diagram on the next page). The recursive put () 
method in algorithm 3.3 accomplishes this task using logic similar to that we Lised for 
the recursive search: if the tree is empty, we return a new node containing the key and 
value; if the search key is less than the key at the root, we set the left link to the result 
of inserting the key into the left subtree; otherwise, we set the right link to the result of 
inserting the key into the right subtree. 
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Recursion. It is worthwhile to take the time to 
understand the dynamics of these recursive im¬ 
plementations. You can think of the code before 
the recursive calls as happening on the way down 
the tree: it compares the given key against the 
key at each node and moves right or left accord¬ 
ingly. Then, think of the code after the recursive 
calls as happening on the way up the tree. For 
get() this amounts to a series of return state¬ 
ments, but for put (), it corresponds to resetting 
the link of each parent to its child on the search 
path and to incrementing the counts on the way 
up the path. In simple BSTs, the only new link 
is the one at the bottom, but resetting the links 
higher up on the path is as easy as the test to 
avoid setting them. Also, we just need to incre¬ 
ment the node count on each node on the path, 
but we use more general code that sets each to 
one plus the sum of the counts in its subtrees. 

Later in this section and in the next section, we 
shall study more advanced algorithms that are 
naturally expressed with this same recursive 

scheme but that can change more links on the search paths and need the more general 
node-count-update code. Elementary BSTs are often implemented with nonrecursive 
code (see exercise 3 . 2 . 12 )—we use recursion in our implementations both to make it 
easy for you to convince yourself that the code is operating as described and to prepare 
the groundwork for more sophisticated algorithms. 



A careful study of the trace for our standard indexing client that is shown on the 
next page will give you a feeling for the way in which binary search trees grow. New 
nodes are attached to null links at the bottom of the tree; the tree structure is not oth¬ 
erwise changed. For example, the root has the first key inserted, one of the children of 
the root has the second key inserted, and so forth. Because each node has two links, the 
tree tends to grow out, rather than just down. Moreover, only the keys on the path from 
the root to the sought or inserted key are examined, so the number of keys examined 
becomes a smaller and smaller fraction of the number of keys in the tree as the tree size 
increases. 
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BST trace for standard indexing client 
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Analysis The running times of algorithms on binary 
search trees depend on the shapes of the trees, which, in turn, 
depend on the order in which keys are inserted. In the best 
case, a tree with N nodes could be perfectly balanced, with ~ lg 
N nodes between the root and each null link. In the worst case 
there could be N nodes on the search path. The balance in typi¬ 
cal trees turns out to be much closer to the best case than the 
worst case. 

For many applications, the following simple model is rea¬ 
sonable: We assume that the keys are (uniformly) random, or, 
equivalently, that they are inserted in random order. Analysis 
of this model stems from the observation that BSTs are dual 
to quicksort. The node at the root of the tree corresponds to 
the first partitioning item in quicksort (no keys to the left are 
larger, and no keys to the right are smaller) and the subtrees are 
built recursively, corresponding to quicksort’s recursive subar¬ 
ray sorts. This observation leads us to the analysis of properties 
of the trees. 




Proposition C. Search hits in a BST built from N random keys require ~ 2 In N 
(about 1.39 IgN) compares, on the average. 

Proof: The number of compares used for a search hit ending at a given node is 
1 plus the depth. Adding the depths of all nodes, we get a quantity known as the 
internal path length of the tree. Thus, the desired quantity is 1 plus the average in¬ 
ternal path length of the BST, which we can analyze with the same argument that 
we used for proposition k in section 2 . 3 : Let C N be the total internal path length 
of a BST built from inserting N randomly ordered distinct keys, so that the average 
cost of a search hit is 1 + C N / N. We have C 0 =Cj = 0 and for N > 1 we can write a 
recurrence relationship that directly mirrors the recursive BST structure: 

C n =N- 1 + (C 0 + C n _ 1 )/N+ (Q + C n _ 2 )/N+ .. . (C N _J + C 0 )/N 

The N — 1 term takes into account that the root contributes 1 to the path length 
of each of the other N — 1 nodes in the tree; the rest of the expression accounts 
for the subtrees, which are equally likely to be any of the N sizes. After rearranging 
terms, this recurrence is nearly identical to the one that we solved in section 2.3 
for quicksort, and we can derive the approximation C N ~ 2N In N. 
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Proposition D. Insertions and search misses in a BST built from N random keys 
require ~ 2 In N (about 1.39 lg N) compares, on the average. 

Proof: Insertions and search misses take one more compare, on the average, than 
search hits. This fact is not difficult to establish by induction (see exercise 3 . 2 . 16 ). 


proposition c says that we should expect the BST search cost for random keys to be 
about 39 percent higher than that for binary search, proposition d says that the extra 
cost is well worthwhile, because the cost of inserting a new key is also expected to be 
logarithmic—flexibility not available with binary search in an ordered array, where the 
number of array accesses required for an insertion is typically linear. As with quicksort, 
the standard deviation of the number of compares is known to be low, so that these 
formulas become increasingly accurate as N increases. 

Experiments. How well does our random-key model match what is found in typical 
symbol-table clients? As usual, this question has to be studied carefully for particular 
practical applications, because of the large potential variation in performance. Fortu¬ 
nately, for many clients, the model is quite good for BSTs. 

For our example study of the cost of the put() operations for FrequencyCounter 
for words of length 8 or more, we see a reduction in the average cost from 484 array 
accesses or compares per operation for Bi narySearchST to 13 for BST, again providing 
a quick validation of the logarithmic performance predicted by the theoretical model. 
More extensive experiments for larger inputs are illustrated in the table on the next 
page. On the basis of propositions c and d, it is reasonable to predict that this number 
should be about twice the natural logarithm of the table size, because the preponder¬ 
ance of operations are searches in a nearly full table. This prediction has at least the 
following inherent inaccuracies: 

■ Many operations are for smaller tables. 

■ The keys are not random. 

■ The table size may be too small for the approximation 2 In N to be accurate. 
Nevertheless, as you can see in the table, this prediction is accurate for our 
FrequencyCounter test cases to within a few compares. Actually, most of the differ¬ 
ence can be explained by refining the mathematics in the approximation (see exercise 
3- 2 -3 5)- 
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scale magnified by a factor of 250 
/ compared to previous figures 



tale.txt leipziglM.txt 



words 

distinct 

compares 

model actual 

words 

distinct 

compares 

model actual 

all words 

135,635 

10,679 

18.6 

17.5 

21,191,455 

534,580 

23.4 

22.1 

8+ letters 

14,350 

5,737 

17.6 

13.9 

4,239,597 

299,593 

22.7 

21.4 

10+ letters 

4,582 

2,260 

15.4 

13.1 

1,610,829 

165,555 

20.5 

19.3 


Average number of compares per put() for FrequencyCounter using BST 
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Order-based methods and deletion An important reason that BSTs are widely 
used is that they allow us to keep the keys in order. As such, they can serve as the basis 
for implementing the numerous methods in our ordered symbol-table API (see page 
366) that allow clients to access key-value pairs not just by providing the key, but also by 
relative key order. Next, we consider implementations of the various methods in our 
ordered symbol-table API. 

Minimum and maximum. If the left link of the root is null, the smallest key in a BST 
is the key at the root; if the left link is not null, the smallest key in the BST is the smallest 
key in the subtree rooted at the node referenced by the left link. This statement is both 
a description of the recursive mi n () method on page 407 and an inductive proof that it 
finds the smallest key in the BST. The computation is equivalent to a simple iteration 
(move left until finding a null link), but we use recursion for consistency. We might 
have the recursive method return a Key instead of a Node, but we will later have a need 
to use this method to access the Node containing 

the minimum key. Finding the maximum key is finding fioor(G) ^ 

similar, moving to the right instead of to the left. 

Floor and ceiling. If a given key key is less than 
the key at the root of a BST, then the floor of key 
(the largest key in the BST less than or equal to 
key) must be in the left subtree. If key is greater 
than the key at the root, then the floor of key 
could be in the right subtree, but only if there is 
a key smaller than or equal to key in the right 
subtree; if not (or if key is equal to the key at the 
root), then the key at the root is the floor of key. 

Again, this description serves both as the basis 
for the recursive f 1 oor O method and for an in¬ 
ductive proof that it computes the desired result. 

Interchanging right and left (and less and greater) 
gives cei 1 i ng(). 

Selection. Selection in a BST works in a man¬ 
ner analogous to the partition-based method of 
selection in an array that we studied in section 
2 . 5 . We maintain in BST nodes the variable N that 
counts the number of keys in the subtree rooted 
at that node precisely to support this operation. 



M) floor(G) must be 
' on the left 
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ALGORITHM 3.3 (continued) Min, max, floor, and ceiling in BSTs 


public Key min() 

{ 

return min(root).key; 

} 

private Node min(Node x) 

{ 

if (x.left == null) return x; 
return min(x.left); 

} 

public Key floor(Key key) 

{ 

Node x = floorfroot, key); 
if (x == null) return null; 
return x.key; 

} 

private Node floor(Node x, Key key) 

{ 

if (x == null) return null; 

int cmp = key.compareTo(x.key); 

if (cmp == 0) return x; 

if (cmp < 0) return floor(x.left, key); 

Node t = f1oor(x.right, key); 

if (t != null) return t; 

else return x; 

} 


Each client method calls a corresponding private method that takes an additional link (to a Node) 
as argument and returns nul 1 or a Node containing the desired Key via the recursive procedure de¬ 
scribed in the text. The max () and cei lingO methods are the same as mi n () and floor() (respec¬ 
tively) with right and left (and < and >) interchanged. 
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Suppose that we seek the key of rank k (the key 


finding sel ect(3) 
the key of rank 3 


such that precisely k other keys in the BST are 
smaller). If the number of keys tin the left sub¬ 
tree is larger than k, we look (recursively) for the 
key of rank k in the left subtree; if t is equal to k, 
we return the key at the root; and if t is smaller 
than k, we look (recursively) for the key of rank 
k — t — 1 in the right subtree. As usual, this de¬ 
scription serves both as the basis for the recursive 
selectO method on the facing page and for a 
proof by induction that it works as expected. 

Rank. The inverse method rank() that returns 



search for key of rank 


return that 
node’s right link 


the rank of a given key is similar: if the given 
key is equal to the key at the root, we return the 
number of keys t in the left subtree; if the given 
key is less than the key at the root, we return the 
rank of the key in the left subtree (recursively 
computed); and if the 
given key is larger than 
the key at the root, we re¬ 
turn t plus one (to count 
the key at the root) plus 
the rank of the key in the 
right subtree (recursively 
computed). 


3-2-1 = 0 on the right 





2 keys in left subtree 
M) so search for key of 
rank 0 on the left 


0 keys in left subtree 
and searching for 
key of rank 0 
so return H 



Selection in a BST 


available for 
garbage collection 


Delete the minimum/maximum. The most difficult BST op¬ 
eration to implement is the deleteO method that removes a 
key-value pair from the symbol table. As a warmup, consider 


update links and node counts 



Deleting the minimum in a BST 


del eteMi n () (remove the key-value pair with the smallest key). 
As with put() we write a recursive method that takes a link to 
a Node as argument and returns a link to a Node, so that we can 
reflect changes to the tree by assigning the result to the link used 
as argument. For del eteMi n() we go left until finding a Node 
that has a null left link and then replace the link to that node by 
its right link (simply by returning the right link in the recursive 
method). The deleted node, with no link now pointing to it, is 
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ALGORITHM 3.3 (continued) Selection and rank in BSTs 


public Key select(int k) 

{ 

return select(root, k).key; 

} 

private Node selectCNode x, int k) 

{ // Return Node containing key of rank k. 

if (x == null) return null; 
int t = size(x.1 eft); 

if (t > k) return select(x.1 eft, k); 

else if (t < k) return select(x.right, k-t-1); 
else return x; 

} 

public int rank(Key key) 

{ return rank(key, root); } 

private int rank(Key key, Node x) 

{ // Return number of keys less than x.key in the subtree rooted at x. 

if (x == null) return 0; 

int cmp = key.compareTo(x.key); 

if (cmp < 0) return rank(key, x.left); 

else if (cmp > 0) return 1 + size(x.left) + rank(key, x.right); 
else return size(x.1 eft); 

} 


This code uses the same recursive scheme that we have been using throughout this chapter to imple¬ 
ment the sel ect() and rank() methods. It depends on using the private si ze() method given at the 
beginning of this section that gives the number of subtrees rooted at each node. 
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available for garbage collection. Our standard recursive setup accomplishes, after the 
deletion, the task of setting the appropriate link in the parent and updating the counts 
in all nodes in the path to the root. The symmetric method works for del eteMaxQ. 


Delete. We can proceed in a similar manner to de- deleting e 
lete any node that has one child (or no children), but 
what can we do to delete a node that has two chil¬ 
dren? We are left with two links, but have a place in 
the parent node for only one of them. An answer to 
this dilemma, first proposed by T. Hibbard in 1962, 
is to delete a node x by replacing it with its successor. 

Because x has a right child, its successor is the node 
with the smallest key in its right subtree. The replace¬ 
ment preserves order in the tree because there are no 
keys between x. key and the successor’s key. We can 
accomplish the task of replacing x by its successor in 
four (!) easy steps: 

■ Save a link to the node to be deleted in t. 

■ Set x to point to its successor mi n (t. ri ght). 

■ Set the right link of x (which is supposed to 
point to the BST containing all the keys larger 
than x. key) to del eteMi n (t. ri ght), the link 
to the BST containing all the keys that are larger 
than x. key after the deletion. 

■ Set the left link of x (which was null) to t. 1 eft 
(all the keys that are less than both the deleted 
key and its successor). 

Our standard recursive setup accomplishes, after the 
recursive calls, the task of setting the appropriate link 
in the parent and decrementing the node counts in 
the nodes on the path to the root (again, we accom¬ 
plish the task of updating the counts by setting the counts in each node on the search 
path to be one plus the sum of the counts in its children). While this method does the 
job, it has a flaw that might cause performance problems in some practical situations. 
The problem is that the choice of using the successor is arbitrary and not symmetric. 
Why not use the predecessor? In practice, it is worthwhile to choose at random between 
the predecessor and the successor. See exercise 3 . 2.42 for details. 




update links and 
node counts after 
recursive calls 


Deletion in a BST 
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ALGORITHM 3.3 (continued) Deletion in BSTs 


public void deleteMinO 

{ 

root = deleteMin(root); 

} 

private Node deleteMin(Node x) 

{ 

if (x.left == null) return x.right; 
x.left = deleteMin(x.1 eft); 
x.N = size(x.left) + size(x.right) + 1; 
return x; 

} 

public void delete(Key key) 

{ root = delete(root, key); } 

private Node delete(Node x, Key key) 

{ 

if (x == null) return null; 
int cmp = key.compareTo(x.key); 

if (cmp < 0) x.left = delete(x.1 eft, key); 

else if (cmp > 0) x.right = delete(x.right, key); 
el se 
{ 

if (x.right == null) return x.left; 
if (x.left == null) return x.right; 

Node t = x; 

x = min(t.right); // See page 407. 

x.right = deleteMin(t.right); 
x.left = t.left; 

} 

x.N = size(x.left) + size(x.right) + 1; 
return x; 

} 


These methods implement eager Hibbard deletion in BSTs, as described in the text on the facing 
page. The delete() code is compact, but tricky. Perhaps the best way to understand it is to read 
the description at left, try to write the code yourself on the basis of the description, then compare 
your code with this code. This method is typically effective, but performance in large-scale applica¬ 
tions can become a bit problematic (see exercise 3 . 2 . 42 ). The del eteMax() method is the same as 
del eteMi n () with right and left interchanged. 
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Range queries. To implement the keys() method that returns the keys in a given 
range, we begin with a basic recursive BST traversal method, known as inorder traversal. 
To illustrate the method, we consider the task of printing all the keys in a BST in order. 
To do so, print all the keys in the left subtree (which are less than the key at the root by 

definition of BSTs), then print the key at the root, then 
print all the keys in the right subtree (which are greater 
than the key at the root by definition of BSTs), as in the 
code at left. As usual, the description serves as an argu¬ 
ment by induction that this code prints the keys in order. 
To implement the two-argument keys () method that re¬ 
turns to a client all the keys in a specified range, we modi¬ 
fy this code to add each key that is in the range to a Queue, 
and to skip the recursive calls for subtrees that cannot 
contain keys in the range. As with Bi narySearchST, the 
fact that we gather the keys in a Queue is hidden from the client. The intention is that 
clients should process all the keys in the range of interest using Java’s foreach construct 
rather than needing to know what data structure we use to implement Iterabl e<Key>. 

Analysis. How efficient are the order-based operations in BSTs? To study this question, 
we consider the tree height (the maximum depth of any node in the tree). Given a tree, 
its height determines the worst-case cost of all BST operations (except for range search 
which incurs additional cost proportional to the number of keys returned). 

Proposition E. In a BST, all operations take time proportional to the height of the 
tree, in the worst case. 

Proof: All of these methods go down one or two paths in the tree. The length of 
any path is no more than the height, by definition. 


private void print(Node x) 
{ 

if (x == null) return; 
pri nt(x. 1 eft) ; 
StdOut.println(x.key) ; 
print(x. right) ; 

} 

Printing the keys in a BST in order 


We expect the tree height (the worst-case cost) to be higher than the average internal 
path length that we defined on page 403 (which averages in the short paths as well), but 
how much higher? This question may seem to you to be similar to the questions an¬ 
swered by proposition c and proposition d, but it is far more difficult to answer, 
certainly beyond the scope of this book. The average height of a BST built from random 
keys was shown to be logarithmic by J. Robson in 1979, and L. Devroye later showed 
that the value approaches 2.99 lg N for large N. Thus, if the insertions in our applica¬ 
tion are well-described by the random-key model, we are well on the way toward our 
goal of developing a symbol-table implementation that supports all of these operations 
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ALGORITHM 3.3 (continued) Range searching in BSTs 


public Iterable<Key> keys() 

{ return keys(min(), max()); } 

public Iterable<Key> keys(Key lo, Key hi) 

{ 

Queue<Key> queue = new Queue<Key>(); 
keys(root, queue, To, hi); 
return queue; 

} 


private void keys(Node x, Queue<Key> queue, Key lo, Key hi) 

{ 


} 


if (x == 
int cmplo 
int cmphi 
if (cmplo 
if (cmplo 
if (cmphi 


null) return; 

= lo.compareTo(x.key); 

= hi.compareTo(x.key); 

< 0) keys(x.left, queue, lo, hi); 

<= 0 && cmphi >= 0) queue.enqueue(x.key) 
> 0) keys(x.right, queue, lo, hi); 


To enqueue all the keys from the tree rooted at a given node that fall in a given range onto a queue, we 
(recursively) enqueue all the keys from the left subtree (if any of them could fall in the range), then 
enqueue the node at the root (if it falls in the range), then (recursively) enqueue all the keys from the 
right subtree (if any of them could fall in the range). 


searching in the range [F. .T] 

red keys are used in compares 
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in logarithmic time. We can expect that no path in a tree built from random keys is 
longer than 3 lg N, but what can we expect if the keys are not random? In the next sec¬ 
tion, you will see why this question is moot in practice because of balanced BSTs, which 
guarantee that the BST height will be logarithmic regardless of the order in which keys 
are inserted. 

in summary, BSTs are not difficult to implement and can provide fast search and insert 
for practical applications of all kinds if the key insertions are well-approximated by the 
random-key model. For our examples (and for many practical applications) BSTs make 
the difference between being able to accomplish the task at hand and not being able to 
do so. Moreover, many programmers choose BSTs for symbol-table implementations 
because they also support fast rank, select, delete, and range query operations. Still, as 
we have emphasized, the bad worst-case performance of BSTs may not be tolerable in 
some situations. Good performance of the basic BST implementation is dependent on 
the keys being sufficiently similar to random keys that the tree is not likely to contain 
many long paths. With quicksort, we were able to randomize; with a symbol-table API, 
we do not have that freedom, because the client controls the mix of operations. Indeed, 
the worst-case behavior is not unlikely in practice—it arises when a client inserts keys 
in order or in reverse order, a sequence of operations that some client certainly might 
attempt in the absence of any explicit warnings to avoid doing so. This possibility is a 
primary reason to seek better algorithms and data structures, which we consider next. 


algorithm 
(data structure) 


worst-case cost average-case cost 

(after N inserts) (after N random inserts) efficiently 

support ordered 

search insert search hit insert operations? 


sequential search 
(unordered linked list) 

N 

N 

N/2 

N 

no 

binary search 
(ordered array ) 

lg N 

N 

lg N 

N/2 

yes 

binary tree search 
(BST) 

N 

N 

1.39 IgN 

1.39 lg N 

yes 


Cost summary for basic symbol-table implementations (updated) 


www.it-ebooks.info 



3.2 ■ Binary Search Trees 


415 


Q&A 


Q. I’ve seen BSTs before, but not using recursion. What are the tradeoffs? 

A. Generally, recursive implementations are a bit easier to verify for correctness; non¬ 
recursive implementations are a bit more efficient. See exercise 3 . 2.13 for an imple¬ 
mentation of get(), the one case where you might notice the improved efficiency. If 
trees are unbalanced, the depth of the function-call stack could be a problem in a recur¬ 
sive implementation. Our primary reason for using recursion is to ease the transition 
to the balanced BST implementations of the next section, which definitely are easier to 
implement and debug with recursion. 

Q. Maintaining the node count field in Node seems to require a lot of code. Is it really 
necessary? Why not maintain a single instance variable containing the number of nodes 
in the tree to implement the si ze() client method? 

A. The rank() and selectO methods need to have the size of the subtree rooted at 
each node. If you are not using these ordered operations, you can streamline the code 
by eliminating this field (see exercise 3 . 2 . 12 ). Keeping the node count correct for all 
nodes is admittedly error-prone, but also a good check for debugging. You might also 
use a recursive method to implement size() for clients, but that would take linear 
time to count all the nodes and is a dangerous choice because you might experience 
poor performance in a client program, not realizing that such a simple operation is so 
expensive. 
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EXERCISES 

3.2.1 Draw the BST that results when you insert the keys E ASYQUESTION, 
in that order (associating the value i with the i th key, as per the convention in the text) 
into an initially empty tree. How many compares are needed to build the tree? 

3.2.2 Inserting the keys in the order A X C S E R H into an initially empty BST gives 
a worst-case tree where every node has one null link, except one at the bottom, which 
has two null links. Give five other orderings of these keys that produce worst-case trees. 

3.2.3 Give five orderings of the keys A X C S E R H that, when inserted into an initially 
empty BST, produce the best-case tree. 

3.2.4 Suppose that a certain BST has keys that are integers between 1 and 10, and we 
search for 5 . Which sequence below cannot be the sequence of keys examined? 


a. 

10, 

9 , 

8, 

7 , 

6, 

5 

b. 

4 , 

10, 

8, 

7 , 

53 


c. 

1, 

10, 

2, 

9 , 

3 , 

8 

d. 

2, 

7 , 

B, 

8, 

4 , 

5 

e. 

1, 

2, 

10, 

4 , 

8, 

5 


3.2.5 Suppose that we have an estimate ahead of time of how often search keys are 
to be accessed in a BST, and the freedom to insert them in any order that we desire. 
Should the keys be inserted into the tree in increasing order, decreasing order of likely 
frequency of access, or some other order? Explain your answer. 

3.2.6 Add to BST a method height () that computes the height of the tree. Develop two 
implementations: a recursive method (which takes linear time and space proportional 
to the height), and a method like si ze 0 that adds a field to each node in the tree (and 
takes linear space and constant time per query). 

3.2.7 Add to BST a recursive method avgCompares () that computes the average num¬ 
ber of compares required by a random search hit in a given BST (the internal path 
length of the tree divided by its size, plus one). Develop two implementations: a re¬ 
cursive method (which takes linear time and space proportional to the height), and a 
method like si ze () that adds a field to each node in the tree (and takes linear space and 
constant time per query). 

3.2.8 Write a static method optCompares () that takes an integer argument N and com¬ 
putes the number of compares required by a random search hit in an optimal (perfectly 
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balanced) BST, where all the null links are on the same level if the number of links is a 
power of 2 or on one of two levels otherwise. 

3.2.9 Draw all the different BST shapes that can result when N keys are inserted into an 
initially empty tree, for N = 2 , B, 4 , 5 , and 6. 

3.2.10 Write a test client TestBST. java for use in testing the implementations of 
min(), max(), floorO, ceilingO, selectO, rank(), deleteO, deleteMinO, 
deleteMaxO, and keys() that are given in the text. Start with the standard index¬ 
ing client given on page 370 . Add code to take additional command-line arguments, as 
appropriate. 

3.2.11 How many binary tree shapes of N nodes are there with height N? How many 
different ways are there to insert N distinct keys into an initially empty BST that result 
in a tree of height N? (See exercise 3.2.2.) 

3.2.12 Develop a BST implementation that omits rank() and sel ect() and does not 
use a count field in Node. 

3.2.13 Give nonrecursive implementations of get() and put() for BST. 

Partial solution: Here is an implementation of get(): 

public Value get(Key key) 

{ 

Node x = root; 
while (x != null) 

{ 

int cmp = key.compareTo(x.key); 
if (cmp == 0) return x.val; 
else if (cmp < 0) x = x.left; 
else if (cmp > 0) x = x.right; 

} 

return nul1; 

} 

The implementation of put () is more complicated because of the need to save a point¬ 
er to the parent node to link in the new node at the bottom. Also, you need a separate 
pass to check whether the key is already in the table because of the need to update the 
counts. Since there are many more searches than inserts in performance-critical imple¬ 
mentations, using this code for get() is justified; the corresponding change for put() 
might not be noticed. 
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EXERCISES (continued) 


3.2.14 Give nonrecursive implementations of min(), max(), floor(), ceilingO, 
rank(),and sel ect(). 

3.2.15 Give the sequences of nodes examined when the methods in BST are used to 
compute each of the following quantities for the tree drawn at right. 

a. floor("Q") 

b. select(5) 

c. ceilingC'Q") 

d. ranker) 

e. sizeC'D", "T") 
f keysC'D", "T") 

3.2.16 Define the external path length of a tree to be the sum of the number of nodes on 
the paths from the root to all null links. Prove that the difference between the external 
and internal path lengths in any binary tree with N nodes is 2 N (see proposition c). 

3.2.17 Draw the sequence of BSTs that results when you delete the keys from the tree 
of exercise 3.2.1, one by one, in the order they were inserted. 

3.2.18 Draw the sequence of BSTs that results when you delete the keys from the tree 
of exercise 3.2.1, one by one, in alphabetical order. 

3.2.19 Draw the sequence of BSTs that results when you delete the keys from the tree 
of exercise 3.2.1, one by one, by successively deleting the key at the root. 

3.2.20 Prove that the running time of the two-argument keys () in a BST with N nodes 
is at most proportional to the tree height plus the number of keys in the range. 

3.2.21 Add a BST method randomKeyO that returns a random key from the symbol 
table in time proportional to the tree height, in the worst case. 

3.2.22 Prove that if a node in a BST has two children, its successor has no left child and 
its predecessor has no right child. 

3.2.23 Is del ete () commutative? (Does deleting x, then y give the same result as de¬ 
leting y, then x?) 

3.2.24 Prove that no compare-based algorithm can build a BST using fewer than 
lg(N!) ~Nig N compares. 
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CREATIVE PROBLEMS 


3.2.25 Perfect balance. Write a program that inserts a set of keys into an initially emp¬ 
ty BST such that the tree produced is equivalent to binary search, in the sense that the 
sequence of compares done in the search for any key in the BST is the same as the se¬ 
quence of compares used by binary search for the same set of keys. 

3.2.26 Exact probabilities. Find the probability that each of the trees in exercise 3 . 2.9 
is the result of inserting N random distinct elements into an initially empty tree. 

3.2.27 Memory usage. Compare the memory usage of BST with the memory usage of 
Bi narySearchST and Sequenti al SearchST for N key-value pairs, under the assump¬ 
tions described in section 1.4 (see exercise 3 . 1 . 21 ). Do not count the memory for 
the keys and values themselves, but do count references to them. Then draw a diagram 
that depicts the precise memory usage of a BST with Stri ng keys and Integer values 
(such as the ones built by FrequencyCounter), and then estimate the memory usage 
(in bytes) for the BST built when FrequencyCounter uses BST for Tale of Two Cities. 

3.2.28 Sofware caching. Modify BST to keep the most recently accessed Node in an 
instance variable so that it can be accessed in constant time if the next put() or get() 
uses the same key (see exercise 3 . 1 . 25 ). 

3.2.29 Binary tree check. Write a recursive method i sBi naryTree() that takes a Node 
as argument and returns true if the subtree count field N is consistent in the data struc¬ 
ture rooted at that node, fal se otherwise. Note: This check also ensures that the data 
structure has no cycles and is therefore a binary tree (!). 

3.2.30 Order check. Write a recursive method i sOrde red O that takes a Node and two 
keys mi n and max as arguments and returns true if all the keys in the tree are between 
mi n and max; mi n and max are indeed the smallest and largest keys in the tree, respec¬ 
tively; and the BST ordering property holds for all keys in the tree; fal se otherwise. 

3.2.31 Equal key check. Write a method hasNoDupli cates () that takes a Node as ar¬ 
gument and returns t rue if there are no equal keys in the binary tree rooted at the argu¬ 
ment node, fal se otherwise. Assume that the test of the previous exercise has passed. 

3.2.32 Certification. Write a method i sBST () that takes a Node as argument and re¬ 
turns true if the argument node is the root of a binary search tree, fal se otherwise. 
Hint: This task is also more difficult than it might seem, because the order in which you 
call the methods in the previous three exercises is important. 
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CREATIVE PROBLEMS (continued) 


Solution: 

private boolean isBSTO 

{ 

if (MsBinaryTree(root)) return false; 
if (!isOrdered(root, min(), max())) return false; 
if (!hasNoDuplicates(root)) return false; 
return true; 

} 

3.2.33 Select/rank check. Write a method that checks, for all i from 0 to size()-l, 
whether i is equal to rank(select(i)) and, for all keys in the BST, whether key is 
equal to select(rank(key)). 

3.2.34 Threading. Your goal is to support an extended API ThreadedST that supports 
the following additional operations in constant time: 


Key next(Key key) key that follows key (null if key is the maximum) 

Key prev(Key key) key that precedes key (null if key is the minimum) 


To do so, add fields pred and succ to Node that contain links to the predecessor and 
successor nodes, and modify put(), deleteMinO, deleteMaxO, and delete() to 
maintain these fields. 

3.2.35 Refined analysis. Refine the mathematical model to better explain the experi¬ 
mental results in the table given in the text. Specifically, show that the average number 
of compares for a successful search in a tree built from random keys approaches the 
limit 2 In N + 2y - 3 = 1.39 lg N - 1.85 as N increases, where y = .57721... is Euler’s 
constant. Hint: Referring to the quicksort analysis in section 2 . 3 , use the fact that the 
integral of l/x approaches In N + y. 

3.2.36 Iterator. Is it possible to write a nonrecursive version of keys () that uses space 
proportional to the tree height (independent of the number of keys in the range)? 

3.2.37 Level-order traversal. Write a method pri ntLevel () that takes a Node as argu¬ 
ment and prints the keys in the subtree rooted at that node in level order (in order of 
their distance from the root, with nodes on each level in order from left to right). Hint: 
Use a Queue. 
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3.2.38 Tree drawing. Add a method draw() to BST that draws BST figures in the style 
of the text. Hint'. Use instance variables to hold node coordinates, and use a recursive 
method to set the values of these variables. 
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EXPERIMENTS 

3.2.39 Average case. Run empirical studies to estimate the average and standard de¬ 
viation of the number of compares used for search hits and for search misses in a BST 
built by running 100 trials of the experiment of inserting N random keys into an ini¬ 
tially empty tree, for N = 10 4 , 10 5 , and 10 s . Compare your results against the formula 
for the average given in exercise 3.2.35. 

3.2.40 Height. Run empirical studies to estimate average BST height by running 100 
trials of the experiment of inserting N random keys into an initially empty tree, for N = 
10 4 , 10 5 , and 10 6 . Compare your results against the 2.99 lg N estimate that is described 
in the text. 

3.2.41 Array representation. Develop a BST implementation that represents the BST 
with three arrays (preallocated to the maximum size given in the constructor): one with 
the keys, one with array indices corresponding to left links, and one with array indices 
corresponding to right links. Compare the performance of your program with that of 
the standard implementation. 

3.2.42 Hibbard deletion degradation. Write a program that takes an integer N from the 
command line, builds a random BST of size N, then enters into a loop where it deletes 
a random key (using the code delete(select(StdRandom. uniform(N)))) and then 
inserts a random key, iterating the loop N 2 times. After the loop, measure and print the 
average length of a path in the tree (the internal path length divided by N, plus 1). Run 
your program for N = 10 2 , 10 3 , and 10 4 to test the somewhat counterintuitive hypoth¬ 
esis that this process increases the average path length of the tree to be proportional to 
the square root of N. Run the same experiments for a del ete() implementation that 
makes a random choice whether to use the predecessor or the successor node. 

3.2.43 Put/get ratio. Determine empirically the ratio of the amount of time that 
BST spends on put() operations to the time that it spends on get() operations when 
FrequencyCounter is used to find the frequency of occurrence of values in 1 million 
randomly-generated integers. 

3.2.44 Cost plots. Instrument BST so that you can produce plots like the ones in this 
section showing the cost of each put() operation during the computation (see exer¬ 
cise 3.1.38). 

3.2.45 Actual timings. Instrument FrequencyCounter to use Stopwatch and StdDraw 
to make a plot where the x axis is the number of calls on get Q or put() and the yaxis 
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is the total running time, with a point plotted of the cumulative time after each call. 
Run your program for Tale of Two Cities using Sequent! al SearchST and again using 
BinarySearchST and again using BST and discuss the results. Note: Sharp jumps in 
the curve may be explained by caching, which is beyond the scope of this question (see 
exercise 3 . 1 . 39 ). 

3.2.46 Crossover to binary search trees. Find the values of N for which using a binary 
search tree to build a symbol table of N random double keys becomes 10, 100, and 
1,000 times faster than binary search. Predict the values with analysis and verify them 
experimentally. 

3.2.47 Average search time. Run empirical studies to compute the average and stan¬ 
dard deviation of the average length of a path to a random node (internal path length 
divided by tree size, plus 1) in a BST built by insertion of N random keys into an initially 
empty tree, for N from 100 to 10,000.Do 1,000 trials for each tree size. Plotthe results in 
a Tufte plot, like the one at the bottom of this page, fit with a curve plotting the function 
1.39 lg N- 1.85 (see exercise 3 . 2.35 and exercise 3 . 2 . 39 ). 



0 


100 


operations 


10000 


Average path length to a random node in a BST built from random keys 
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3.3 BALANCED SEARCH TREES 



The algorithms in the previous section work well for a wide variety of applications, but 
they have poor worst-case performance. We introduce in this section a type of binary 
search tree where costs are guaranteed to be logarithmic, no matter what sequence of 
keys is used to construct them. Ideally, we would like to keep our binary search trees 
perfectly balanced. In an /V-node tree, we would like the height to be ~lg N so that we 
can guarantee that all searches can be completed in ~lg N compares, just as for binary 
search (see proposition b). Unfortunately, maintaining perfect balance for dynamic 
insertions is too expensive. In this section, we consider a data structure that slightly re¬ 
laxes the perfect balance requirement to provide guaranteed logarithmic performance 
not just for the insert and search operations in our symbol-table API but also for all of 
the ordered operations (except range search). 


2-3 search trees The primary step to get the flexibility that we need to guarantee 
balance in search trees is to allow the nodes in our trees to hold more than one key. Spe¬ 
cifically, referring to the nodes in a standard BST as 2-nodes (they hold two links and 
one key), we now also allow 3-nodes, which hold three links and two keys. Both 2-nodes 
and 3-nodes have one link for each of the intervals subtended by its keys. 



Anatomy of a 2-3 search tree 


Definition. A 2-3 search tree is a tree that is either empty or 

■ A 2-node, with one key (and associated value) and two links, 
a left link to a 2-3 search tree with smaller keys, and a right 
link to a 2-3 search tree with larger keys 

■ A 3-node, with two keys (and associated values) and three 
links, a left link to a 2-3 search tree with smaller keys, a mid¬ 
dle link to a 2-3 search tree with keys between the node’s 
keys, and a right link to a 2-3 search tree with larger keys 

As usual, we refer to a link to an empty tree as a null link. 


A perfectly balanced 2-3 search tree is one whose null links are all the same distance 
from the root. To be concise, we use the term 2-3 tree to refer to a perfectly balanced 2-3 
search tree (the term denotes a more general structure in other contexts). Later, we shall 
see efficient ways to define and implement the basic operations on 2-nodes, 3-nodes, 
and 2-3 trees; for now, let us assume that we can manipulate them conveniently and see 
how we can use them as search trees. 
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successful search for H 


unsuccessful search for B 


H is less than M so 



B is less than M so 
look to the left 




B is between A and C so look in the middle 
link is null so B is not in the tree (search miss ) 


Search hit (left) and search miss (right) in a 2-3 tree 


Search. The search algorithm for keys in a 2-3 tree directly generalizes the search al¬ 
gorithm for BSTs. To determine whether a key is in the tree, we compare it against the 
keys at the root. If it is equal to any of them, we have a search hit; otherwise, we follow 
the link from the root to the subtree corresponding to the interval of key values that 
could contain the search key. If that link is null, we have a search miss; otherwise we 
recursively search in that subtree. 



\ replace 2-node with 
new 3-node containing K 

Insert into a 2-node 


Insert into a 2-node. To insert a new node in a 2-3 
tree, we might do an unsuccessful search and then 
hook on the node at the bottom, as we did with 
BSTs, but the new tree would not remain perfectly 
balanced. The primary reason that 2-3 trees are 
useful is that we can do insertions and still main¬ 
tain perfect balance. It is easy to accomplish this 
task if the node at which the search terminates is 
a 2-node: we just replace the node with a 3-node 
containing its key and the new key to be inserted. If 
the node where the search terminates is a 3-node, 
we have more work to do. 
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inserting S 

'S 1 ?- 



■ no room for S 


<— make a 4-node 

split 4-node into 
^ this 2-3 tree 


inserting Z 


Insert into a tree consisting of a single 3-node. As a first warmup before considering 
the general case, suppose that we want to insert into a tiny 2-3 tree consisting of just a 
single 3-node. Such a tree has two keys, but no room for a new key in its one node. To be 
able to perform the insertion, we temporarily put the new key into a 4-node, a natural 
extension of our node type that has three keys and four links. Creating the 4-node is 
convenient because it is easy to convert it into a 2-3 tree made up of three 2-nodes, one 
with the middle key (at the root), one with the smallest of 
the three keys (pointed to by the left link of the root), and 
one with the largest of the three keys (pointed to by the 
right link of the root). Such a tree is a 3-node BST and also 
a perfectly balanced 2-3 search tree, with all the null links 
at the same distance from the root. Before the insertion, the 
height of the tree is 0; after the insertion, the height of the 
tree is 1. This case is simple, but it is worth considering be¬ 
cause it illustrates height growth in 2-3 trees. Insert into a single 3-node 

Insert into a 3-node whose parent is a 2-node. As a second warmup, suppose that the 
search ends at a 3-node at the bottom whose parent is a 2-node. In this case, we can still 
make room for the new key while maintaining perfect balance in the tree, by making a 

temporary 4-node as just described, then splitting the 
4-node as just described, but then, instead of creat¬ 
ing a new node to hold the middle key, moving the 
middle key to the node’s parent. You can think of the 
transformation as replacing the link to the old 3-node 
in the parent by the middle key with links on either 
side to the new 2-nodes. By our assumption, there 
is room for doing so in the parent: the parent was a 
2-node (with one key and two links) and becomes 
a 3-node (with two keys and three links). Also, this 
transformation does not affect the defining properties 
of (perfectly balanced) 2-3 trees. The tree remains or¬ 
dered because the middle key is moved to the parent, 
and it remains perfectly balanced: if all null links are 
the same distance from the root before the insertion, 
they are all the same distance from the root after the 
insertion. Be certain that you understand this trans¬ 
formation—it is the crux of 2-3 tree dynamics. 



split 4-node into two 2-nodes 
pass middle key to parent 


Insert into a 3-node whose parent is a 2-node 
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Insert into a 3-node whose parent is a 3-node. Now 
suppose that the search ends at a node whose parent is 
a 3-node. Again, we make a temporary 4-node as just 
described, then split it and insert its middle key into 
the parent. The parent was a 3-node, so we replace it 
with a temporary new 4-node containing the middle 
key from the 4-node split. Then, we perform precisely 
the same transformation on that node. That is, we split 
the new 4-node and insert its middle key into its par¬ 
ent. Extending to the general case is clear: we con¬ 
tinue up the tree, splitting 4-nodes and inserting their 
middle keys in their parents until reaching a 2-node, 
which we replace with a 3-node that does not need to 
be further split, or until reaching a 3-node at the root. 




Splitting the root. If we have 3-nodes along the 
whole path from the insertion point to the root, we 

end up with a tempo- 


split 4-node into two 2-nodes 
pass middle key to parent 

add middle key E to 2-node 


inserting D 

search for D ends 
at this 3-node 



add new key D to 3-node 
to make temporary 4-node 




E 3 
H) 


add middle key C to 3-node 
to make temporary 4-node 

Y 



C E 


Y 


x 

(AJ (D) 

\ / 

split 4-node into two 2-nodes 
pass middle key to parent 


split 4-node into 
three 2-nodes 
increasing tree 
height by 1 




split 4-node into two 2-nodes 
pass middle key to parent 

Insert into a 3-node whose parent is a 3-node 


Splitting the root 


rary 4-node at the root. 

In this case we can pro¬ 
ceed in precisely the 
same way as for inser¬ 
tion into a tree consist¬ 
ing of a single 3-node. 

We split the tempo¬ 
rary 4-node into three 

2-nodes, increasing the height of the tree by 1. Note that 
this last transformation preserves perfect balance be¬ 
cause it is performed at the root. 

Local transformations. Splitting a temporary 4-node 
in a 2-3 tree involves one of six transformations, sum¬ 
marized at the bottom of the next page. The 4-node may 
be the root; it may be the left child or the right child of a 
2-node; or it may be the left child, middle child, or right 
child of a 3-node. The basis of the 2-3 tree insertion al¬ 
gorithm is that all of these transformations are purely lo¬ 
cal: no part of the tree needs to be examined or modified 
other than the specified nodes and links. The number of 
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links changed for each trans¬ 
formation is bounded by a 
small constant. In particular, 
the transformations are effec¬ 
tive when we find the specified 
patterns anywhere in the tree, 
not just at the bottom. Each of 
the transformations passes up 
one of the keys from a 4-node 
to that node’s parent in the tree 
and then restructures links ac¬ 
cordingly, without touching 
any other part of the tree. 

Global properties. Moreover, 
these local transformations 
preserve the global properties that the tree is ordered and perfectly balanced: the num¬ 
ber of links on the path from the root to any null link is the same. For reference, a com¬ 
plete diagram illustrating this point for the case that the 4-node is the middle child of a 
3-node is shown above. If the length of every path from a root to a null link is h before 
the transformation, then it is h after the transformation. Each transformation preserves 
this property, even while splitting the 4-node into two 2-nodes and while changing the 
parent from a 2-node to a 3-node or from a 3-node into a temporary 4-node. When 
the root splits into three 2 -nodes, the length of every path from the root to a null link 
increases by 1. If you are not fully convinced, work exercise 3 . 3 . 7 , which asks you to 



Splitting a 4-node is a local transformation 
that preserves order and perfect balance 


root parent is a 3-node 



Splitting a temporary 4-node in a 2-3 tree (summary) 
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extend the diagrams at the top of the previous page for the other five cases to illustrate 
the same point. Understanding that every local transformation preserves order and 
perfect balance in the whole tree is the key to understanding the algorithm. 

unlike standard bsts, which grow down from the top, 2-3 trees grow up from the 
bottom. If you take the time to carefully study the figure on the next page, which gives 
the sequence of 2-3 trees that is produced by our standard indexing test client and the 
sequence of 2-3 trees that is produced when the same keys are inserted in increasing or¬ 
der, you will have a good understanding of the way that 2-3 trees are built. Recall that in 
a BST, the increasing-order sequence for 10 keys results in a worst-case tree of height 9. 
In the 2-3 trees, the height is 2. 

The preceding description is sufficient to define a symbol-table implementation 
with 2-3 trees as the underlying data structure. Analyzing 2-3 trees is different from 
analyzing BSTs because our primary interest is in worst-case performance, as opposed 
to average-case performance (where we analyze expected performance under the ran¬ 
dom-key model). In symbol-table implementations, we normally have no control over 
the order in which clients insert keys into the table and worst-case analysis is one way 
to provide performance guarantees. 

Proposition F. Search and insert operations in a 2-3 tree with N keys are guaran¬ 
teed to visit at most lg N nodes. 

Proof: The height of an N-node 2-3 tree is between [log 3 N\ = [(lg N)/(lg 3)J (if 
the tree is all 3-nodes) and [lg iVj (if the tree is all 2-nodes) (see exercise 3 . 3 . 4 ). 


Thus, we can guarantee good worst-case performance with 2-3 trees. The amount of 
time required at each node by each of the operations is bounded by a constant, and 
both operations examine nodes on just one path, so the total cost of any search or insert 
is guaranteed to be logarithmic. As you can see from comparing the 2-3 tree depicted 
at the bottom of page 431 with the BST formed from the same keys on page 405, a perfectly 
balanced 2-3 tree strikes a remarkably flat posture. For example, the height of a 2-3 
tree that contains 1 billion keys is between 19 and 30. It is quite remarkable that we can 
guarantee to perform arbitrary search and insertion operations among 1 billion keys by 
examining at most 30 nodes. 

However, we are only part of the way to an implementation. Although it is possible 
to write code that performs transformations on distinct data types representing 2 - and 
3-nodes, most of the tasks that we have described are inconvenient to implement in 
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2-3 construction traces 
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this direct representation because there are numerous different cases to be handled. 
We would need to maintain two different types of nodes, compare search keys against 
each of the keys in the nodes, copy links and other information from one type of node 
to another, convert nodes from one type to another, and so forth. Not only is there a 
substantial amount of code involved, but the overhead incurred could make the algo¬ 
rithms slower than standard BST search and insert. The primary purpose of balancing 
is to provide insurance against a bad worst case, but we would prefer the overhead cost 
for that insurance to be low. Fortunately, as you will see, we can do the transformations 
in a uniform way using little overhead. 
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Red-black BSTs The insertion algorithm for 2-3 trees just described is not difficult 
to understand; now, we will see that it is also not difficult to implement. We will con¬ 
sider a simple representation known as a red-black BST that leads to a natural imple¬ 
mentation. In the end, not much code is required, but understanding how and why the 
code gets the job done requires a careful look. 

Encoding 3-nodes. The basic idea behind red-black 
BSTs is to encode 2-3 trees by starting with standard 
BSTs (which are made up of 2-nodes) and adding extra 
information to encode 3-nodes. We think of the links 
as being of two different types: red links, which bind 
together two 2-nodes to represent 3-nodes, and black 
links, which bind together the 2-3 tree. Specifically, 
we represent 3-nodes as two 2-nodes connected by a 
single red link that leans left (one of the 2-nodes is the 
left child of the other). One advantage of using such a 
representation is that it allows us to use our get() code 
for standard BST search without modification. Given 
any 2-3 tree, we can immediately derive a corresponding BST, just by converting each 
node as specified. We refer to BSTs that represent 2-3 trees in this way as red-black BSTs. 

An equivalent definition. Another way to proceed is to define red-black BSTs as BSTs 
having red and black links and satisfying the following three restrictions: 

■ Red links lean left. 

■ No node has two red links connected to it. 

■ The tree has perfect black balance: every path from the root to a null link has the 
same number of black links. 

There is a 1 -1 correspondence between red-black BSTs defined in this way and 2-3 trees. 

A 1-1 correspondence. If we draw the red links horizontally in a red-black BST, all of 
the null links are the same distance from the root, and if we then collapse together the 
nodes connected by red links, the result is a 2-3 tree. Conversely, if we draw 3-nodes in 



Encoding a 3-node with two 2-nodes 
connected by a left-leaning red link 
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1-1 correspondence between red-black BSTs and 2-3 trees 

Color representation. For convenience, since 
each node is pointed to by precisely one link 
(from its parent), we encode the color of links 
in nodes, by adding a bool ean instance variable 
color to our Node data type, which is true if 
the link from the parent is red and fal se if tit 
is black. By convention, null links are black. 
For clarity in our code, we define constants 
RED and BLACK for use in setting and testing 
this variable. We use a private method i sRed () 
to test the color of a node’s link to its parent. 
When we refer to the color of a node, we are 
referring to the color of the link pointing to it, 
and vice versa. 

Rotations. The implementation that we will 
consider might allow right-leaning red links or 
two red links in a row during an operation, but 
it always corrects these conditions before com¬ 
pletion, through judicious use of an operation 
called rotation that switches the orientation of 


a 2-3 tree as two 2-nodes connected by 
a red link that leans left, then no node 
has two red links connected to it, and 
the tree has perfect black balance, since 
the black links correspond to the 2-3 
tree links, which are perfectly balanced 
by definition. Whichever way we choose 
to define them, red-black BSTs are both 
BSTs and 2-3 trees. Thus, if we can im¬ 
plement the 2-3 tree insertion algorithm 
by maintaining the 1-1 correspondence, 
then we get the best of both worlds: the 
simple and efficient search method from 
standard BSTs and the efficient inser¬ 
tion-balancing method from 2-3 trees. 



private static final boolean RED = true; 
private static final boolean BLACK = false; 

private class Node 
{ 

Key key; // key 

Value val; // associated data 

Node left, right; // subtrees 

int N; // # nodes in this subtree 

boolean color; // color of link from 

// parent to this node 

NodeCKey key, Value val, int N, boolean color) 
{ 

this.key = key; 

this.val = val; 

this.N = N; 

this.color = color; 

1 

} 

private boolean isRed(Node x) 

{ 

if (x == null) return false; 
return x.color == RED; 


Node representation for red-black BSTs 
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red links. First, suppose that we have a right-leaning red link that 
needs to be rotated to lean to the left (see the diagram at left). This 
operation is called a left rotation. We organize the computation as 
a method that takes a link to a red-black BST as argument and, as¬ 
suming that link to be to a Node h whose right link is red, makes the 
necessary adjustments and returns a link to a node that is the root of 
a red-black BST for the same set of keys whose left link is red. If you 
check each of the lines of code against the before/affer drawings in 
the diagram, you will find this operation is easy to understand: we 
are switching from having the smaller of the two keys at the root to 
having the larger of the two keys at the root. Implementing a right 
rotation that converts a left-leaning red link to a right-leaning one 
amounts to the same code, with left and right interchanged (see the 
diagram at right below). 

Resetting the link in the parent after a rotation. Whether left or 
right, every rotation leaves us with a link. We always use the link 
returned by rotateRi ght() or rotateLeftO to reset the appro¬ 
priate link in the parent (or the root of 
the tree). That may be a right or a left 
link, but we can always use it to reset 
the link in the parent. This link may be 
red or black—both rotateLeftO and 
rotateRi ght() preserve its color by setting x. col or to 
h. color. This might allow two red links in a row to occur 
within the tree, but our algorithms will also use rotations 
to correct this condition when it arises. For example, the 
code 

h = rotateLeft(h); 

rotates left a right-leaning red link that is to the right of 
node h, setting h to point to the root of the resulting sub¬ 
tree (which contains all the same nodes as the subtree 
pointed to by h before the rotation, but a different root). 

The ease of writing this type of code is the primary reason 
we use recursive implementations of BST methods, as it 
makes doing rotations an easy supplement to normal in¬ 
sertion, as you will see. 



Node rotateRight(Node h) 

{ 

Node x = h.left; 
h.left = x.right; 
x. right = h; 
x.color = h.color; 
h.color = RED; 
x.N = h.N; 

h.N = 1 + size(h.left) 

+ size(h.right) ; 


return x; 



could be right or left, 



{ 

Node x = h.right; 
h.right = x.1 eft; 
x.left = h; 
x.color = h.color; 
h.color = RED; 
x.N = h.N; 

h.N = 1 + size (h.left) 

+ size(h.right); 


return x; 



Left rotate (right link of h) 


Right rotate (left link of h) 
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we can use rotations to help maintain the 1-1 correspondence between 
2-3 trees and red-black BSTs as new keys are inserted because they pre¬ 
serve the two defining properties of red-black BSTs: order and perfect black 
balance. That is, we can use rotations on a red-black BST without having 
to worry about losing its order or its perfect black balance. Next, we see 
how to use rotations to preserve the other two defining properties of red- 
black BSTs (no consecutive red links on any path and no right-leaning red 
links). We warm up with some easy cases. 

Insert into a single 2-node. A red-black BST with 1 key is just a single 
2-node. Inserting the second key immediately shows the need for having 
a rotation operation. If the new key is smaller than the key in the tree, we 
just make a new (red) node with the new key and we are done: we have 
a red-black BST that is equivalent to a single 3-node. But if the new key 
is larger than the key in the tree, then attaching a new (red) node gives a 
right-leaning red link, and the code root = rotateLeft(root) ; com¬ 
pletes the insertion by rotating the red link to the left and updating the 
tree root link. The result in both cases is the red-black representation of a 
single 3-node, with two keys, one left-leaning red link, and black height 1. 
Insert into a 2-node at the bottom. We insert keys into a red-black BST 
as usual into a BST, adding a new node at the bottom (respecting the or¬ 
der), but always connected to its parent with a red link. If the parent is a 
2-node, then the same two cases just discussed are effective. If the new node 
is attached to the left link, the parent simply becomes a 3-node; if it is at¬ 
tached to a right link, we have a 3-node leaning the wrong way, but a left 
rotation finishes the job. 

Insert into a tree with two keys (in a 3-node). This case reduces to three 
subcases: the new key is either less than both keys in the tree, between them, 
or greater than both of them. Each of the cases introduces a node with two 
red links connected to it; our goal is to correct this condition. 

■ The simplest of the three cases is when the new key is larger than 
the two in the tree and is therefore attached on the rightmost link of 
the 3-node, making a balanced tree with the middle key at the root, 
connected with red links to nodes containing a smaller and a larger 
key. If we flip the colors of those two links from red to black, then we 
have a balanced tree of height 2 with three nodes, exactly what we 
need to maintain our 1-1 correspondence to 2-3 trees. The other two 
cases eventually reduce to this case. 


root 


search ends 
at this null link 

root 

b ) red link to 
new node 
containing a 
converts 2-node 
to 3-node 



right 


^ root 

search ends 
this null link 

/CJ\ attached new node 
' with red link 

root 

r§) 

\ rotated left 
yV to make a 
' ' legal 3-node 


Insert into a single 
2-node (two cases) 


insert C 



node here 


right link red 



Insert into a 2-node 
at the bottom 
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■ If the new key is 
smaller than the two 
keys in the tree and 
goes on the left link, 
then we have two 
red links in a row, 
both leaning to the 
left, which we can 
reduce to the previ¬ 
ous case (middle 
key at the root, con¬ 
nected to the others 
by two red links) by 
rotating the top link 
to the right. 

■ If the new key goes 
between the two 
keys in the tree, we 


larger 


smaller 


between 



search ends 
^ at this 
null link 



Xs '" search ends 
at this null link 



& A 


search ends 
fK^at this null link 



zps colors flipped 
(hi to black 


Insert into a single 3-node (three cases) 



void fiipCoiors(Node h) 
{ 


h.color = RED; 
h.left.color = BLACK; 
h.right.col or = BLACK; 



Flipping colors to split a 4-node 


again have two red links in a row, a right-leaning one below 
a left-leaning one, which we can reduce to the previous case 
(two red links in a row, to the left) by rotating left the bot¬ 
tom link. 

In summary, we achieve the desired result by doing zero, 
one, or two rotations followed by flipping the colors of the 
two children of the root. As with 2-3 trees, be certain that 
you understand these transformations, as they are the key to 
red-black tree dynamics. 

Flipping colors. To flip the colors of the two red children 
of a node, we use a method fl i pCol ors(), shown at left. In 
addition to flipping the colors of the children from red to 
black, we also flip the color of the parent from black to red. 
A critically important characteristic of this operation is that, 
like rotations, it is a local transformation that preserves per¬ 
fect black balance in the tree. Moreover, this convention im¬ 
mediately leads us to a full implementation, as we describe 
next. 
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/ 

add new 
node here 


Keeping the root black. In the case just considered (insert into a single 3-node), the 
color flip will color the root red. This can also happen in larger trees. Strictly speaking, 
a red root implies that the root is part of a 3-node, but that is not the case, so we color 
the root black after each insertion. Note that the black 
height of the tree increases by 1 whenever the color of 
the color of the root is flipped from black to red. 

Insert into a 3-node at the bottom. Now suppose that 
we add a new node at the bottom that is connected to a 
3-node. The same three cases just discussed arise. Either 
the new link is connected to the right link of the 3-node 
(in which case we just flip colors) or to the left link of 
the 3-node (in which case we need to rotate the top link 
right and flip colors) or to the middle link of the 3-node 
(in which case we rotate left the bottom link, then rotate 
right the top link, then flip colors). Flipping the colors 
makes the link to the middle node red, which amounts 
to passing it up to its parent, putting us back in the same 
situation with respect to the parent, which we can fix by 
moving up the tree. 

Passing a red link up the tree. The 2-3 tree insertion 
algorithm calls for us to split the 3-node, passing the 
middle key up to be inserted into its parent, continuing 
until encountering a 2-node or the root. In every case 
we have considered, we precisely accomplish this objec¬ 
tive: after doing any necessary rotations, we flip colors, 
which turns the middle node to red. From the point of 
view of the parent of that node, that link becoming red 
can be handled in precisely the same manner as if the 
red link came from attaching a new node: we pass up a 
red link to the middle node. The three cases summa¬ 
rized in the diagram on the next page precisely capture 

the operations necessary in a red-black tree to implement the key operation in 2-3 tree 
insertion: to insert into a 3-node, create a temporary 4-node, split it, and pass a red link 
to the middle key up to its parent. Continuing the same process, we pass a red link up 
the tree until reaching a 2-node or the root. 


two lefts in a row 



right link red 
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IN SUMMARY, WE CAN MAINTAIN OUT 1-1 
correspondence between 2-3 trees and 
red-black BSTs during insertion by judi¬ 
cious use of three simple operations: left 
rotate, right rotate, and color flip. We can 
accomplish the insertion by performing 
the following operations, one after the 
other, on each node as we pass up the tree 
from the point of insertion: 

■ If the right child is red and the left 
child is black, rotate left. 

■ If both the left child and its left 
child are red, rotate right. 

■ If both children are red, flip colors. 

It certainly is worth your while to check that this sequence handles each of the cases 
just described. Note that the first operation handles both the rotation necessary to lean 
the 3-node to the left when the parent is a 2-node and the rotation necessary to lean the 
bottom link to the left when the new red link is the middle link in a 3-node. 

Implementation Since the balancing operations are to be performed on the way 
up the tree from the point of insertion, implementing them is easy in our standard 
recursive implementation: we just do them after the recursive calls, as shown in algo¬ 
rithm 3.4. The three operations listed in the previous paragraph each can be accom¬ 
plished with a single i f statement that tests the colors of two nodes in the tree. Even 
though it involves a small amount of code, this implementation would be quite difficult 
to understand without the two layers of abstraction that we have developed (2-3 trees 
and red-black BSTs) to implement it. At a cost of testing three to five node colors (and 
perhaps doing a rotation or two or flipping colors when a test succeeds), we get BSTs 
that have nearly perfect balance. 

The traces for our standard indexing client and for the same keys inserted in increas¬ 
ing order are given on page 440. Considering these examples simply in terms of our 
three operations on red-black trees, as we have been doing, is an instructive exercise. 
Another instructive exercise is to check the correspondence with 2-3 trees that the algo¬ 
rithm maintains (using the figure for the same keys given on page 430). In both cases, 
you can test your understanding of the algorithm by considering the transformations 
(two color flips and two rotations) that are needed when P is inserted into the red-black 
BST (see exercise 3.3.12). 
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ALGORITHM 3.4 Insert for red-black BSTs 


public class RedBlackBST<Key extends Comparable<Key>, Value> 

{ 

private Node root; 


private class Node // BST node with color bit (see page 433) 


private boolean isRed(Node h) // See page 433. 
private Node rotateLeft(Node h) // See page 434. 
private Node rotateRight(Node h) // See page 434. 
private void flipColors(Node h) // See page 436. 

private int sizeQ // See page 398. 


public void put(Key key, Value val) 

{ // Search for key. Update value if found; grow table if new. 

root = putfroot, key, val); 
root.color = BLACK; 


private Node put(Node h, Key key, Value val) 

{ 

if (h == null) // Do standard insert, with red link to parent, 
return new Node(key, val, 1, RED); 


int cmp = key.compareTofh.key); 

if (cmp < 0) h.left = put(h.left, key, val); 

else if (cmp > 0) h.right = put(h.right, key, val); 
else h.val = val; 


if (isRed(h. right) && HsRed(h.left)) h = rotateLeft(h) ; 
if (isRed(h.1 eft) && isRedfh.left.left)) h = rotateRight(h); 
if (isRed(h.1 eft) && isRed(h.right)) flipColors(h); 

h.N = size(h.left) + size(h.right) + 1; 
return h; 

} 

} 


The code for the recursive put() for red-black BSTs is identical to put() in elementary BSTs except 
for the three i f statements after the recursive calls, which provide near-perfect balance in the tree 
by maintaining a 1-1 correspondence with 2-3 trees, on the way up the search path. The first rotates 
left any right-leaning 3-node (or a right-leaning red link at the bottom of a temporary 4-node); the 
second rotates right the top link in a temporary 4-node with two left-leaning red links; and the third 
flips colors to pass a red link up the tree (see text). 
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Red-black BST construction traces 
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Deletion Since put() in algorithm 3.4 is already one of the most intricate 
methods that we consider in this book, and the implementations of deleteMinO, 
deleteMaxO, and deletef) for red-black BSTs are a bit more complicated, we defer 
their full implementations to exercises. Still, the basic approach is worthy of study. To 
describe it, we begin by returning to 2-3 trees. As with insertion, we can define a se¬ 
quence of local transformations that allow us to delete a node while still maintaining 
perfect balance. The process is somewhat more complicated than for insertion, because 
we do the transformations both on the way down the search path, 
when we introduce temporary 4-nodes (to allow for a node to be 
deleted), and also on the way up the search path, where we split any 
leftover 4-nodes (in the same manner as for insertion). 

Top-down 2-3-4 trees. As a first warmup for deletion, we con¬ 
sider a simpler algorithm that does transformations both on the 
way down the path and on the way up the path: an insertion algo¬ 
rithm for 2-3-4 trees, where the temporary 4-nodes that we saw in 
2-3 trees can persist in the tree. The insertion algorithm is based on 
doing transformations on the way down the path to maintain the 
invariant that the current node is not a 4-node (so we are assured 
that there will be room to insert the new key at the bottom) and 
transformations on the way up the path to balance any 4-nodes 
that may have been created. The transformations on the way down 
are precisely the same transformations that we used for splitting 
4-nodes in 2-3 trees. If the root is a 4-node, we split it into three 

2- nodes, increasing the height of the tree by 1. On the way down 
the tree, if we encounter a 4-node with a 2-node as parent, we split 
the 4-node into two 2-nodes and pass the middle key to the par¬ 
ent, making it a 3-node; if we encounter a 4-node with a 3-node as 
parent, we split the 4-node into two 2-nodes and pass the middle 
key to the parent, making it a 4-node. We do not need to worry 
about encountering a 4-node with a 4-node as parent by virtue of 
the invariant. At the bottom, we have, again by virtue of the invariant, a 2-node or a 

3- node, so we have room to insert the new key. To implement this algorithm with red- 
black BSTs, we 

■ Represent 4-nodes as a balanced subtree of three 2-nodes, with both the left and 
right child connected to the parent with a red link 

■ Split 4-nodes on the way down the tree with color flips 

■ Balance 4-nodes on the way up the tree with rotations, as for insertion 


at the root 



on the way down 



at the bottom 

« — « 
« — PP3 

Transformations for insert 
in top-down 2-3-4 trees 


www.it-ebooks.info 























442 


CHAPTER 3 ■ Searching 


Remarkably, you can implement top-down 2-3-4 trees by moving one line of code in 
put() in algorithm 3 . 4 : move the colorFl ip() call (and accompanying test) to be¬ 
fore the recursive calls (between the test for null and the comparison). This algorithm 
has some advantages over 2-3 trees in applications where multiple processes have access 
to the same tree, because it always is operating within a link or two of the current node. 
The deletion algorithms that we describe next are based on a similar scheme and are 
effective for these trees as well as for 2-3 trees. 

Delete the minimum. As a second warmup 
for deletion, we consider the operation of 
deleting the minimum from a 2-3 tree. The 
basic idea is based on the observation that we 
can easily delete a key from a 3-node at the 
bottom of the tree, but not from a 2 -node. 

Deleting the key from a 2-node leaves a node 
with no keys; the natural thing to do would 
be to replace the node with a null link, but 
that operation would violate the perfect bal¬ 
ance condition. So, we adopt the following 
approach: to ensure that we do not end up on 
a 2 -node, we perform appropriate transfor¬ 
mations on the way down the tree to preserve 
the invariant that the current node is not a 
2-node (it might be a 3-node or a tempo¬ 
rary 4-node). First, at the root, there are two 
possibilities: if the root is a 2 -node and both 
children are 2 -nodes, we can just convert the 
three nodes to a 4-node; otherwise we can 
borrow from the right sibling if necessary to ensure that the left child of the root is not 
a 2-node. Then, on the way down the tree, one of the following cases must hold: 

■ If the left child of the current node is not a 2-node, there is nothing to do. 

■ If the left child is a 2-node and its immediate sibling is not a 2-node, move a key 
from the sibling to the left child. 

■ If the left child and its immediate sibling are 2-nodes, then combine them with 
the smallest key in the parent to make a 4-node, changing the parent from a 
3-node to a 2-node or from a 4-node to a 3-node. 

Following this process as we traverse left links to the bottom, we wind up on a 3-node 
or a 4-node with the smallest key, so we can just remove it, converting the 3-node to a 


at the root 





PPP PP 

Transformations for delete the minimum 
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2-node or the 4-node to a 3-node. Then, on the way up the tree, we split any unused 
temporary 4-nodes. 

Delete. The same transformations along the search path just described for deleting the 
minimum are effective to ensure that the current node is not a 2-node during a search 
for any key. If the search key is at the bottom, we can just remove it. If the key is not 
at the bottom, then we have to exchange it with its successor as in regular BSTs. Then, 
since the current node is not a 2-node, we have reduced the problem to deleting the 
minimum in a subtree whose root is not a 2-node, and we can use the procedure just 
described for that subtree. After the deletion, as usual, we split any remaining 4-nodes 
on the search path on the way up the tree. 

several of the exercises at the end of this section are devoted to examples and 
implementations related to these deletion algorithms. People with an interest in devel¬ 
oping or understanding implementations need to master the details covered in these 
exercises. People with a general interest in the study of algorithms need to recognize 
that these methods are important because they represent the first symbol-table imple¬ 
mentation that we have seen where search, insert, and delete are all guaranteed to be 
efficient, as we will establish next. 
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Properties of red-black BSTs Studying the properties of red-black BSTs is a 
matter of checking the correspondence with 2-3 trees and then applying the analysis of 
2-3 trees. The end result is that all symbol-table operations in red-black BSTs are guaran¬ 
teed to be logarithmic in the size of the tree (except for range search, which additionally 
costs time proportional to the number of keys returned). We repeat and emphasize this 
point because of its importance. 

Analysis. First, we establish that red-black BSTs, while not perfectly balanced, are al¬ 
ways nearly so, regardless of the order in which the keys are inserted. This fact immedi¬ 
ately follows from the 1-1 correspondence with 2-3 trees and the defining property of 
2-3 trees (perfect balance). 

Proposition G. The height of a red-black BST with N nodes is no more than 2 lg N. 

Proof sketch: The worst case is a 2-3 tree that is all 2-nodes except that the leftmost 
path is made up of 3-nodes. The path taking left links from the root is twice as long 
as the paths of length ~ lg N that involve just 2-nodes. It is possible, but not easy, to 
develop key sequences that cause the construction of red-black BSTs whose average 
path length is the worst-case 2 lg N. If you are mathematically inclined, you might 
enjoy exploring this issue by working exercise 3 . 3 . 24 . 


This upper bound is conservative: experiments involving both random insertions and 
insertion sequences found in typical applications support the hypothesis that each 
search in a red-black BST of N nodes uses about 1.00 lg AT- .5 compares, on the aver¬ 
age. Moreover, you are not likely to encounter a substantially higher average number of 
compares in practice. 



Typical red-black BST built from random keys (null links omitted) 
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tal e. 

txt 



leipziglM. 

. txt 



words 

distinct 

compares 
model actual 

words 

distinct 

compares 

model actual 

all words 

135,635 

10,679 

13.6 

13.5 

21,191,455 

534,580 

19.4 

19.1 

8+ letters 

14,350 

5,737 

12.6 

12.1 

4,239,597 

299,593 

18.7 

18.4 

10+ letters 

4,582 

2,260 

11.4 

11.5 

1,610,829 

165,555 

17.5 

17.3 


Average number of compares per put() for FrequencyCounter using RedBlackBST 


Property H. The average length of a path from the root to a node in a red-black 
BST with N nodes is -1.00 IgN. 

Evidence: Typical trees, such as the one at the bottom of the previous page (and 
even the one built by inserting keys in increasing order at the bottom of this page) 
are quite well-balanced, by comparison with typical BSTs (such as the tree depicted 
on page 405). The table at the top of this page shows that path lengths (search costs) 
for our FrequencyCounter application are about 40 percent lower than from el¬ 
ementary BSTs, as expected. This performance has been observed in countless ap¬ 
plications and experiments since the invention of red-black BSTs. 


For our example study of the cost of the put() operations for FrequencyCounter for 
words of length 8 or more, we see a further reduction in the average cost, again pro¬ 
viding a quick validation of the logarithmic performance predicted by the theoretical 
model, though this validation is less surprising than for BSTs because of the guarantee 
provided by property g. The total savings is less than the 40 per cent savings in the 
search cost because we count rotations and color flips as well as compares. 
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The get() method in red-black BSTs does not examine the node color, so the balanc¬ 
ing mechanism adds no overhead; search is faster than in elementary BSTs because 
the tree is balanced. Each key is inserted just once, but may be involved in many, many 
search operations, so the end result is that we get search times that are close to optimal 
(because the trees are nearly balanced and no work for balancing is done during the 
searches) at relatively little cost (unlike binary search, insertions are guaranteed to be 
logarithmic). The inner loop of the search is a compare followed by updating a link, 
which is quite short, like the inner loop of binary search (compare and index arithme¬ 
tic). This implementation is the first we have seen with logarithmic guarantees for both 
search and insert, and it has a tight inner loop, so its use is justified in a broad variety of 
applications, including library implementations. 

Ordered symbol-table API. One of the most appealing features of red-black BSTs is 
that the complicated code is limited to the put O (and deletion) methods. Our code for 
the minimum/maximum, select, rank, floor, ceiling and range queries in standard BSTs 
can be used without any change, since it operates on BSTs and has no need to refer to the 
node color, algorithm 3 . 4 , together with these methods (and the deletion methods), 
leads to a complete implementation of our ordered symbol-table API. Moreover, all of 
the methods benefit from the near-perfect balance in the tree because they all require 
time proportional to the tree height, at most. Thus proposition g, in combination 
with proposition e, suffices to establish a logarithmic performance guarantee for all 
of them. 
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Proposition I. In a red-black BST, the following operations take logarithmic time 
in the worst case: search, insertion, finding the minimum, finding the maximum, 
floor, ceiling, rank, select, delete the minimum, delete the maximum, delete, and 
range count. 

Proof: We have just discussed get(), put(), and the deletion operations. For the 
others, the code from section 3.2 can be used verbatim (it just ignores the node 
color). Guaranteed logarithmic performance follows from propositions e and G, 
and the fact that each algorithm performs a constant number of operations on each 
node examined. 


On reflection, it is quite remarkable that we are able to achieve such guarantees. In a 
world awash with information, where people maintain tables with trillions or quadril¬ 
lions of entries, the fact is that we can guarantee to complete any one of these opera¬ 
tions in such tables with just a few dozen compares. 


algorithm 
(data structure) 

worst-case cost 
(after N inserts) 

search insert 

average-case cost 
(after N random inserts) 

search hit insert 

efficiently 
support ordered 
operations? 

sequential search 
(unordered linked list) 

N 

N 

N/2 

N 

no 

binary search 
(ordered array) 

lg N 

N 

lg N 

N/2 

yes 

binary tree search 
(BST) 

N 

N 

1.39 lg N 

1.39 lg N 

yes 

2-3 tree search 
(red-black BST) 

2 lg N 

2 lg N 

1.00 lg N 

1.00 lg N 

yes 


Cost summary for symbol-table implementations (updated) 
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| Q&A 

Q. Why not let the 3-nodes lean either way and also allow 4-nodes in the trees? 

A. Those are fine alternatives, used by many for decades. You can learn about several of 
these alternatives in the exercises. The left-leaning convention reduces the number of 
cases and therefore requires substantially less code. 

Q. Why not use an array of Key values to represent 2-, 3-, and 4-nodes with a single 
Node type? 

A. Good question. That is precisely what we do for B-trees (see chapter 6), where we 
allow many more keys per node. For the small nodes in 2-3 trees, the overhead for the 
array is too high a price to pay. 

Q. When we split a 4-node, we sometimes set the color of the right node to RED in 
rotateRightO and then immediately set it to BLACK in flipColorsC). Isn’t that 
wasteful? 

A. Yes, and we also sometimes unnecessarily recolor the middle node. In the grand 
scheme of things, resetting a few extra bits is not in the same league with the improve¬ 
ment from linear to logarithmic that we get for all operations, but in performance-crit¬ 
ical applications, you can put the code for rotateRightO and flipColorsO inline 
and eliminate those extra tests. We use those methods for deletion, as well, and find 
them slightly easier to use, understand, and maintain by making sure that they preserve 
perfect black balance. 
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EXERCISES 


3 . 3.1 Draw the 2-3 tree that results when you insert the keys E ASYQUTIONin 
that order into an initially empty tree. 


3 . 3.2 Draw the 2-3 tree that results when you insert the keys YLPMXHCRAES 
in that order into an initially empty tree. 


3 . 3.3 Find an insertion order for the keys SEARCHXM that leads to a 2-3 tree 
of height 1 . 


3 . 3.4 Prove that the height of a 2-3 tree with A' keys is between ~ [log 3 iVj 
.63 lg N (for a tree that is all 3-nodes) and ~[lg N\ (for a tree that is all 

2 -nodes). 

3 . 3.5 The figure at right shows all the structurally different 2-3 trees with N 
keys, for N from 1 up to 6 (ignore the order of the subtrees). Draw all the 
structurally different trees for N= 7, 8 , 9, and 10. 


R 




3 . 3.6 Find the probability that each of the 2-3 trees in exercise 3 . 3.5 is the 
result of the insertion of N random distinct keys into an initially empty tree. 

3 . 3.7 Draw diagrams like the one at the top of page 428 for the other five cases 
in the diagram at the bottom of that page. 

3 . 3.8 Show all possible ways that one might represent a 4-node with three 
2 -nodes bound together with red links (not necessarily left-leaning). 




3 . 3.9 Which of the following are red-black BSTs? 





3 . 3.10 Draw the red-black BST that results when you insert items with the keys 
EASYQUTIONin that order into an initially empty tree. 

3 . 3.11 Draw the red-black BST that results when you insert items with the keys 
YLPMXHCRAES in that order into an initially empty tree. 
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EXERCISES (continued) 


3 . 3.12 Draw the red-black BST that results after each transformation (color flip or 
rotation) during the insertion of P for our standard indexing client. 

3 . 3.13 True or false: If you insert keys in increasing order into a red-black BST, the tree 
height is monotonically increasing. 

3 . 3.14 Draw the red-black BST that results when you insert letters A through K in order 
into an initially empty tree, then describe what happens in general when trees are built 
by insertion of keys in ascending order (see also the figure in the text). 

3 . 3.15 Answer the previous two questions 
for the case when the keys are inserted in de¬ 
scending order. 

3 . 3 . 16 Show the result of inserting n into the 
red-black BST drawn at right (only the search 
path is shown, and you need to include only 
these nodes in your answer). 

3 . 3.17 Generate two random 16-node red- 
black BSTs. Draw them (either by hand or 
with a program). Compare them with the 
(unbalanced) BSTs built with the same keys. 

3 . 3 . 18 Draw all the structurally different red-black BSTs with N keys, for N from 2 up 
to 10 (see exercise 3 . 3 . 5 ). 

3 . 3.19 With 1 bit per node for color, we can represent 2-, 3-, and 4-nodes. How many 
bits per node would we need to represent 5-, 6 -, 7-, and 8 -nodes with a binary tree? 

3 . 3.20 Compute the internal path length in a perfectly balanced BST of N nodes, when 
N is a power of 2 minus 1. 

3 . 3.21 Create a test client TestRB . java, based on your solution to exercise 3 . 2 . 10 . 

3 . 3.22 Find a sequence of keys to insert into a BST and into a red-black BST such that 
the height of the BST is less than the height of the red-black BST, or prove that no such 
sequence is possible. 
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CREATIVE PROBLEMS 


3.3.23 2-3 trees without balance restriction. Develop an implementation of the basic 
symbol-table API that uses 2-3 trees that are not necessarily balanced as the underlying 
data structure. Allow 3-nodes to lean either way. Hook the new node onto the bottom 
with a black link when inserting into a 3-node at the bottom. Run experiments to de¬ 
velop a hypothesis estimating the average path length in a tree built from N random 
insertions. 

3.3.24 Worst case for red-black BSTs. Show how to construct a red-black BST dem¬ 
onstrating that, in the worst case, almost all the paths from the root to a null link in a 
red-black BST of N nodes are of length 2 lg N. 

3.3.25 Top-down 2-3-4 trees. Develop an implementation of the basic symbol-table 
API that uses balanced 2-3-4 trees as the underlying data structure, using the red-black 
representation and the insertion method described in the text, where 4-nodes are split 
by flipping colors on the way down the search path and balancing on the way up. 

3.3.26 Single top-down pass. Develop a modified version of your solution to exer¬ 
cise 3 . 3.25 that does not use recursion. Complete all the work splitting and balancing 
4-nodes (and balancing 3-nodes) on the way down the tree, finishing with an insertion 
at the bottom. 

3.3.27 Allow right-leaning red links. Develop a modified version of your solution to 
exercise 3 . 3.25 that allows right-leaning red links in the tree. 

3.3.28 Bottom-up 2-3-4 trees. Develop an implementation of the basic symbol-table 
API that uses balanced 2-3-4 trees as the underlying data structure, using the red-black 
representation and a bottom-up insertion method based on the same recursive approach 
as algorithm 3 . 4 . Your insertion method should split only the sequence of 4-nodes (if 
any) on the bottom of the search path. 

3.3.29 Optimal storage. Modify RedBl ackBST so that it does not use any extra storage 
for the color bit, based on the following trick: To color a node red, swap its two links. 
Then, to test whether a node is red, test whether its left child is larger than its right child. 
You have to modify the compares to accommodate the possible link swap, and this trick 
replaces bit compares with key compares that are presumably more expensive, but it 
shows that the bit in the nodes can be eliminated, if necessary. 

3.3.30 Sofware caching. Modify RedBl ackBST to keep the most recently accessed Node 
in an instance variable so that it can be accessed in constant time if the next put() or 
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CREATIVE PROBLEMS (continued) 


get() uses the same key (see exercise 3 . 1 . 25 ). 

3.3.31 Tree drawing. Add a method draw() to RedBlackBST that draws red-black 
BST figures in the style of the text (see exercise 3 . 2 . 38 ) 

3.3.32 AVL trees. An AVL tree is a BST where the height of every node and that of 
its sibling differ by at most 1. (The oldest balanced tree algorithms are based on using 
rotations to maintain height balance in AVL trees.) Show that coloring red links that 
go from nodes of even height to nodes of odd height in an AVL tree gives a (perfectly 
balanced) 2-3-4 tree, where red links are not necessarily left-leaning. Extra credit: De¬ 
velop an implementation of the symbol-table API that uses this as the underlying data 
structure. One approach is to keep a height field in each node, using rotations after the 
recursive calls to adjust the height as necessary; another is to use the red-black represen¬ 
tation and use methods like moveRedLeftO and moveRedRightO in exercise 3 . 3.39 
and exercise 3 . 3 . 40 . 

3.3.33 Certification. Add to RedBl ackBST a method i s2 3 0 to check that no node is 
connected to two red links and that there are no right-leaing red links and a method 
i sBal anced () to check that all paths from the root to a null link have the same number 
of black links. Combine these methods with code from i sBST () in exercise 3 . 2.32 to 
create a method i sRedBl ackBST () that checks that the tree is a red-black BST. 

3.3.34 All 2-3 trees. Write code to generate all structurally different 2-3 trees of height 
2, 3, and 4. There are 2, 7, and 122 such trees, respectively. (Hint: Use a symbol table.) 

3.3.35 2-3 trees. Write a program TwoThreeST. java that uses two node types to im¬ 
plement 2-3 search trees directly. 

3.3.36 2-3-4-5-6-7-8 trees. Describe algorithms for search and insertion in balanced 
2-3-4-5-6-7-8 search trees. 

3.3.37 Memoryless. Show that red-black BSTs are not memoryless: for example, if you 
insert a key that is smaller than all the keys in the tree and then immediately delete the 
minimum, you may get a different tree. 

3.3.38 Fundamental theorem of rotations. Show that any BST can be transformed into 
any other BST on the same set of keys by a sequence of left and right rotations. 
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3.3.39 Delete the minimum. Implement the deleteMinO operation for red-black 
BSTs by maintaining the correspondence with the transformations given in the text for 
moving down the left spine of the tree while maintaining the invariant that the current 
node is not a 2-node. 

Solution: 

private Node moveRedLeft(Node h) 

{ // Assuming that h is red and both h.left and h.left.left 

// are black, make h.left or one of its children red. 
flipColors(h)I 
if (isRed(h.right.left)) 

{ 

h.right = rotateRight(h.right); 
h = rotateLeft(h); 

} 

return h; 

} 

public void deleteMinO 

{ 

if (!isRedfroot.1 eft) && !isRed(root.right)) 
root.color = RED; 
root = deleteMin(root); 
if (lisEmptyO) root.color = BLACK; 

} 

private Node deleteMin(Node h) 

{ 

if (h.left == null) 
return nul1; 

if (!isRed(h.1 eft) && !isRed(h.left.1 eft)) 
h = moveRedLeft(h); 
h.left = deleteMin(h.left); 
return balance(h); 

} 

This code assumes a bal ance() method that consists of the line of code 
if (isRed(h.right)) h = rotateLeft(h); 


www.it-ebooks.info 


454 


CHAPTER 3 ■ Searching 


CREATIVE PROBLEMS (continued) 


followed by the last five lines of the recursive put() in algorithm 3.4 and a 
flipColorsO implementation that complements the three colors, instead of the 
method given in the text for insertion. For deletion, we set the parent to BLACK and the 
two children to RED. 

3.3.40 Delete the maximum. Implement the deleteMaxO operation for red-black 
BSTs. Note that the transformations involved differ slightly from those in the previous 
exercise because red links are left-leaning. 

Solution: 

private Node moveRedRight(Node h) 

{ // Assuming that h is red and both h.right and h.right.left 

// are black, make h.right or one of its children red. 
flipColors(h) 
if (!isRed(h.1 eft.left)) 
h = rotateRight(h); 
return h; 

} 

public void deleteMaxO 

{ 

if (!isRed(root.1 eft) && !isRedfroot.right)) 
root.color = RED; 
root = deleteMax(root); 
if (lisEmptyO) root, col or = BLACK; 

} 

private Node deleteMax(Node h) 

{ 

if (isRedfh.1 eft)) 

h = rotateRight(h); 
if (h.right == nul1) 
return nul1; 

if (!isRed(h.right) && !isRed(h.right.1 eft)) 
h = moveRedRight(h); 
h.right = deleteMaxfh.right); 
return balance(h); 

} 
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3.3.41 Delete. Implement the del ete () operation for red-black BSTs, combining the 
methods of the previous two exercises with the del ete() operation for BSTs. 

Solution: 

public void deletefKey key) 

{ 

if (!isRedfroot.1 eft) && !isRed(root.right)) 
root.color = RED; 
root = delete(root, key); 
if (HsEmptyO) root.color = BLACK; 

} 

private Node delete(Node h, Key key) 

{ 

if (key.compareTo(h.key) < 0) 

{ 

if (!isRed(h.1 eft) && !isRed(h.left.left)) 
h = moveRedLeft(h); 
h.left = delete(h.1 eft, key); 

} 

el se 

{ 

if (isRed(h.left)) 
h = rotateRight(h); 

if (key.compareTofh.key) == 0 && (h.right == null)) 
return nul1; 

if (!isRed(h.right) && !isRed(h.right.1 eft)) 
h = moveRedRight(h); 
if (key.compareTo(h.key) == 0) 

{ 

h.val = get(h.right, min(h.right).key); 
h.key = min(h.right).key; 
h.right = deleteMin(h.right); 

} 

else h.right = delete(h.right, key); 

} 

return balance(h); 

} 
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EXPERIMENTS 

3.3.42 Count red nodes. Write a program that computes the percentage of red nodes 
in a given red-black BST. Test your program by running at least 100 trials of the experi¬ 
ment of inserting N random keys into an initially empty tree, for N = 10 4 , 10 5 , and 10 6 , 
and formulate an hypothesis. 

3.3.43 Cost plots. Instrument RedBlackBST so that you can produce plots like the 
ones in this section showing the cost of each put() operation during the computation 
(see exercise 3 . 1 . 38 ). 

3.3.44 Average search time. Run empirical studies to compute the average and stan¬ 
dard deviation of the average length of a path to a random node (internal path length 
divided by tree size) in a red-black BST built by insertion of N random keys into an 
initially empty tree, for N from 1 to 10,000. Do at least 1,000 trials for each tree size. 
Plot the results in a Tufte plot, like the one at the bottom of this page, fit with a curve 
plotting the function lg N- .5. 

3.3.45 Count rotations. Instrument your program for exercise 3 . 3.43 to plot the 
number of rotations and node splits that are used to build the trees. Discuss the results. 

3.3.46 Height. Instrument your program for exercise 3 . 3.43 to plot the height of 
red-black BSTs. Discuss the results. 
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Average path length to a random node in a red-black BST built from random keys 
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3.4 HASH TABLES 



If keys are small integers, we can use an array to implement an unordered symbol table, 
by interpreting the key as an array index so that we can store the value associated with 
key i in array entry i, ready for immediate access. In this section, we consider hashing, 
an extension of this simple method that handles more complicated types of keys. We 
reference key-value pairs using arrays by doing arithmetic operations to transform keys 
into array indices. 

Search algorithms that use hashing consist of two separate parts. The first part is to 
compute a hash function that transforms the search key into an array index. Ideally, 
different keys would map to different indices. This ideal is gener¬ 
ally beyond our reach, so we have to face the possibility that two 
or more different keys may hash to the same array index. Thus, 
the second part of a hashing search is a collision-resolution process 
that deals with this situation. After describing ways to compute 
hash functions, we shall consider two different approaches to col¬ 
lision resolution: separate chaining and linear probing. 

Hashing is a classic example of a time-space tradeoff. If there 
were no memory limitation, then we could do any search with 
only one memory access by simply using the key as an index in 
a (potentially huge) array. This ideal often cannot be achieved, 
however, because the amount of memory required is prohibitive 
when the number of possible key values is huge. On the other 
hand, if there were no time limitation, then we can get by with 
only a minimum amount of memory by using sequential search 
in an unordered array. Hashing provides a way to use a reasonable 
amount of both memory and time to strike a balance between 
these two extremes. Indeed, it turns out that we can trade off time 
and memory in hashing algorithms by adjusting parameters, not by rewriting code. To 
help choose values of the parameters, we use classical results from probability theory. 

Probability theory is a triumph of mathematical analysis that is beyond the scope of 
this book, but the hashing algorithms we consider that take advantage of the knowl¬ 
edge gained from that theory are quite simple, and widely used. With hashing, you can 
implement search and insert for symbol tables that require constant (amortized) time 
per operation in typical applications, making it the method of choice for implementing 
basic symbol tables in many situations. 
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Hash functions The first problem that we face is the computation of the hash 
function, which transforms keys into array indices. If we have an array that can hold 
M key-value pairs, then we need a hash function that can transform any given key into 
an index into that array: an integer in the range [0, M- 1], We seek a hash function 
that both is easy to compute and uniformly distributes the keys: for each key, every 
integer between 0 and M- 1 should be equally likely (independently for 
every key). This ideal is somewhat mysterious; to understand hashing, 
it is worthwhile to begin by thinking carefully about how to implement 
such a function. 

The hash function depends on the key type. Strictly speaking, we need 
a different hash function for each key type that we use. If the key involves a 
number, such as a social security number, we could start with that num¬ 
ber; if the key involves a string, such as a person’s name, we need to con¬ 
vert the string into a number; and if the key has multiple parts, such as 
a mailing address, we need to combine the parts somehow. For many 
common types of keys, we can make use of default implementations pro¬ 
vided by Java. We briefly discuss potential implementations for various 
types of keys so that you can see what is involved because you do need to 
provide implementations for key types that you create. 

Typical example. Suppose that we have an application where the keys 
are U.S. social security numbers. A social security number such as 
123-45-6789 is a nine-digit number divided into three fields. The first 
field identifies the geographical area where the number was issued (for 
example, social security numbers whose first field is 035 are from Rhode 
Island and numbers whose first field is 214 are from Maryland) and the 
other two fields identify the individual. There are a billion (10 9 ) different 
social security numbers, but suppose that our application will need to 
process just a few hundred keys, so that we could use a hash table of size 
M = 1,000. One possible approach to implementing a hash function is 
to use three digits from the key. Using three digits from the third field is 
likely to be preferable to using the three digits in the first field (since customers may not 
be uniformly dispersed over geographic areas), but a better approach is to use all nine 
digits to make an i nt value, then consider hash functions for integers, described next. 

Positive integers. The most commonly used method for hashing integers is called 
modular hashing: we choose the array size M to be prime and, for any positive inte¬ 
ger key k, compute the remainder when dividing k by M. This function is very easy to 
compute (k % M, in Java) and is effective in dispersing the keys evenly between 0 and 
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M- 1. If M is not prime, it may be the case that not all of the bits of the key play a role, 
which amounts to missing an opportunity to disperse the values evenly. For example, 
if the keys are base-10 numbers and M is 10 k , then only the k least significant digits are 
used. As a simple example where such a choice might be problematic, suppose that the 
keys are telephone area codes and M = 100. For historical reasons, most area codes in 
the United States have middle digit 0 or 1, so this choice strongly favors the values less 
than 20, where the use of the prime value 97 better disperses them (a prime value not 
close to 100 would do even better). Similarly, IP addresses that are used in the internet 
are binary numbers that are not random for similar historical reasons as for telephone 
area codes, so we need to use a table size that is a prime (in particular, not a power of 2) 
if we want to use modular hashing to disperse them. 

Floating-point numbers. If the keys are real numbers between 0 and 1, we might just 
multiply by M and round off to the nearest integer to get an index between 0 and M- 1. 
Although this approach is intuitive, it is defective because it gives more weight to the 
most significant bits of the keys; the least significant bits play no role. One way to ad¬ 
dress this situation is to use modular hashing on the binary representation of the key 
(this is what Java does). 

Strings. Modular hashing works for long keys such as strings, too: we simply treat 
them as huge integers. For example, the code at left computes a modular hash func¬ 
tion for a Stri ng s: recall that charAtO returns a char value in Java, which is a 16-bit 
nonnegative integer. If R is greater than any character value, this computation would 

be equivalent to treating the String as 
an N-digit base-R integer, computing the 
remainder that results when dividing that 
number by M. A classic algorithm known 
as Horner’s method gets the job done with 
N multiplications, additions, and modulus 
operations. If the value of R is sufficiently 
small that no overflow occurs, the result is an integer between 0 and M -1, as desired. 
The use of a small prime integer such as 31 ensures that the bits of all the characters 
play a role. Java’s default implementation for Stri ng uses a method like this. 

Compound keys. If the key type has multiple integer fields, we can typically mix them 
together in the way just described for Stri ng values. For example, suppose that search 
keys are of type Date, which has three integer fields: day (two-digit day), month (two- 
digit month), and year (four-digit year).We compute the number 

int hash = (((day * R + month) % M ) * R + year) % M; 


int hash = 0; 

for (int i =0; i < s.length(); i++) 
hash = (R * hash + s.charAt(i)) % M; 

Hashing a string key 
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which, if the value of R is sufficiently small that no overflow occurs, is an integer be¬ 
tween 0 and M -1, as desired. In this case, we could save the cost of the inner % M opera¬ 
tion by choosing a moderate prime value such as 31 for R. As with strings, this method 
generalizes to handle any number of fields. 

Java conventions. Java helps us address the basic problem that every type of data needs 
a hash function by ensuring that every data type inherits a method called hashCodeO 
that returns a 32-bit integer. The implementation of hashCodeO for a data type must 
be consistent with equals. That is, if a. equal s (b) is true, then a. hashCode () must have 
the same numerical value as b. hashCode (). Conversely, if the hashCodeO values are 
different, then we know that the objects are not equal. If the hashCodeO values are 
the same, the objects may or may not be equal, and we must use equals() to decide 
which condition holds. This convention is a basic requirement for clients to be able to 
use hashCodeO for symbol tables. Note that it implies that you must override both 
hashCodeO and equal s() if you need to hash with a user-defined type. The default 
implementation returns the machine address of the key object, which is seldom what 
you want. Java provides hashCodeO implementations that override the defaults for 
many common types (including Stri ng, Integer, Doubl e, Fi 1 e, and URL). 

Converting a hashCodeO to an array index. Since our goal is an array index, not a 
32-bit integer, we combine hashCode () with modular hashing in our implementations 
to produce integers between 0 and M-1, as follows: 

private int hash(Key x) 

{ return (x.hashCodeO & 0x7fffffff) % M; } 

This code masks off the sign bit (to turn the 32-bit number into a 31-bit nonnegative 
integer) and then computes the remainder when dividing by M, as in modular hashing. 
Programmers commonly use a prime number for the hash table size M when using code 


like this, to attempt to make use of all the 
bits of the hash code. Note: To avoid con¬ 
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hash values in the table at right. 


Hash values for keys in examples 



User-defined hashCodeO. Client code expects that hashCodeO disperses the keys 
uniformly among the possible 32-bit result values. That is, for any object x, you can 
write x. hashCodeO and, in principle, expect to get any one of the 2 32 possible 32-bit 
values with equal likelihood. Java’s hashCodeO implementations for Stri ng, Integer, 
Double, File, and URL aspire to this functionality; for your own type, you have to 
try to do it on your own. The Date example that we considered on page 460 illustrates 
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one way to proceed: make integers from 
the instance variables and use modular 
hashing. In Java, the convention that all 
data types inherit a hashCodeO method 
enables an even simpler approach: use the 
hashCodeO method for the instance vari¬ 
ables to convert each to a 32-bit i nt value 
and then do the arithmetic, as illustrated 
at left for Transaction. For primitive- 
type instance variables, note that a cast to 
a wrapper type is necessary to access the 
hashCodeO method. Again, the precise 
values of the multiplier (31 in our exam¬ 
ple) is not particularly important. 

Software caching. If computing the hash 
code is expensive, it may be worthwhile to 
cache the hash for each key. That is, we maintain an instance variable hash in the key 
type that contains the value of hashCodeO for each key object (see exercise 3 . 4 . 25 ). 
On the first call to hashCodeO, we have to compute the full hash code (and set the val¬ 
ue of hash), but subsequent calls on hashCodeO simply return the value of hash. Java 
uses this technique to reduce the cost of computing hashCodeO for Stri ng objects. 

in summary, we have three primary requirements in implementing a good hash 
function for a given data type: 

■ It should be consistent —equal keys must produce the same hash value. 

■ It should be efficient to compute. 

■ It should uniformly distribute the keys. 

Satisfying these requirements simultaneously is a job for experts. As with many built-in 
capabilities, Java programmers who use hashing assume that hashCodeO does the job, 
absent any evidence to the contrary. 

Still, you should be vigilant whenever using hashing in situations where good perfor¬ 
mance is critical, because a bad hash function is a classic example of a performance bug: 
everything will work properly, but much more slowly than expected. Perhaps the easiest 
way to ensure uniformity is to make sure that all the bits of the key play an equal role in 
computing every hash value; perhaps the most common mistake in implementing hash 
functions is to ignore significant numbers of the key bits. Whatever the implementa¬ 
tion, it is wise to test any hash function that you use, when performance is important. 
Which takes more time: computing a hash function or comparing two keys? Does your 


public class Transaction 
{ 

private final String who; 
private final Date when; 
private final double amount; 

public int hashCodeO 
{ 

int hash = 17; 

hash = 31 * hash + who.hashCodeO; 
hash = 31 * hash + when.hashCodeO; 
hash = 31 * hash 

+ ((Double) amount) . hashCodeO ; 
return hash; 

} 

} 

Implementing hashCodeO in a user-defined type 
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110 » 10679/97 



Hash value frequencies for words in Tale of Two Cities (10,679 keys, M = 97) 


hash function spread a typical set of keys uniformly among the values between 0 and 
M- 1? Doing simple experiments that answer these questions can protect future clients 
from unfortunate surprises. For example, the histogram above shows that our hash() 
implementation using the hashCodeO from Java’s String data type produces a rea¬ 
sonable dispersion of the words for our Tale of Two Cities example. 

Underlying this discussion is a fundamental assumption that we make when using 
hashing, an idealized model that we do not actually expect to achieve, but that guides 
our thinking when implementing hashing algorithms: 

Assumption J (uniform hashing assumption). The hash functions that we use uni¬ 
formly and independently distribute keys among the integer values between 0 and 
M-l. 

Discussion: With all of the arbitrary choices that we have made, we certainly do not 
have hash functions that uniformly and independently distribute keys in this strict 
mathematical sense. Indeed, the idea of implementing consistent functions that are 
guaranteed to uniformly and independently distribute keys leads to deep theoreti¬ 
cal studies that tell us that computing such a function easily is likely to be a very 
elusive goal. In practice, as with random numbers generated by Math. random(), 
most programmers are content to have hash functions that cannot easily be distin¬ 
guished from random ones. Few programmers check for independence, however, 
and this property is rarely satisfied. 


DESPITE THE DIFFICULTY OF VALIDATING IT, ASSUMPTION J is a useful Way to think 
about hashing for two primary reasons. First, it is a worthy goal when designing hash 
functions that guides us away from making arbitrary decisions that might lead to an 
excessive number of collisions. Second, even though we may not be able to validate the 
assumption itself, it does enable us to use mathematical analysis to develop hypotheses 
about the performance of hashing algorithms that we can check with experiments. 
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Hashing with separate chaining A hash function converts keys into array in¬ 
dices. The second component of a hashing algorithm is collision resolution: a strategy 
for handling the case when two or more keys to be inserted hash to the same index. A 
straightforward and general approach to collision resolution is to build, for each of the 
M array indices, a linked list of the key-value pairs whose keys hash to that index. This 
method is known as separate chaining because items that collide are chained together 
in separate linked lists. The basic idea is to choose M to be sufficiently large that the lists 
are sufficiently short to enable efficient search through a two-step process: hash to find 
the list that could contain the key, then sequentially search through that list for the key. 

One way to proceed is to ex¬ 
pand SequentialSearchST (al¬ 
gorithm 3 . 1 ) to implement sep¬ 
arate chaining using linked-list 
primitives (see exercise 3 . 4 . 2 ). 
A simpler (though slightly less 
efficient) way to proceed is to 
adopt a more general approach: 
we build, for each of the M ar¬ 
ray indices, a symbol table of the 
keys that hash to that index, thus 
reusing code that we have already 
developed. The implementa¬ 
tion SeparateChainingHashST 
in algorithm 3.5 maintains an 
array of SequentialSearchST 
objects and implements getO 
and put() by computing a 
hash function to choose which 
Sequenti al SearchST object can contain the key and then using get() and put() (re¬ 
spectively) from Sequential SearchST to complete the job. 

Since we have M lists and N keys, the average length of the lists is always N/M, no 
matter how the keys are distributed among the lists. For example, suppose that all the 
items fall onto the first list—the average length of the lists is (N+ 0 + 0 + 0 +... + 0)/M = 
N/M. However the keys are distributed on the lists, the sum of the list lengths is N and 
the average is N/M. Separate chaining is useful in practice because each list is extremely 
likely to have about N/M key-value pairs. In typical situations, we can verify this con¬ 
sequence of assumption j and count on fast search and insert. 
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Hashing with separate chaining for standard indexing client 
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ALGORITHM 3.5 Hashing with separate chaining 


public class SeparateChainingHashSTcKey, Value> 

{ 

private int N; // number of key-value pairs 

private int M; // hash table size 

private SequentialSearchST<Key, Value>[] st; // array of ST objects 

public SeparateChai ni ngHashSTO 
{ this(997); } 

public SeparateChainingHashST(int M) 

{ // Create M linked lists. 

this.M = M; 

st = (SequentialSearchST<Key, Value>[]) new SequentialSearchST[M]; 
for (int i =0; i < M; i++) 

st [i ] = new Sequenti al SearchSTO ; 

} 

private int hash(Key key) 

{ return (key.hashCode() & 0x7fffffff) % M; } 
public Value get(Key key) 

{ return (Value) st[hash(key)].get(key); } 

public void put(Key key, Value val) 

{ st[hash(key)].put(key, val); } 

public Iterable<Key> keys() 

// See Exercise 3.4.19. 

} 


This basic symbol-table implementation maintains an array of linked lists, using a hash function to 
choose a list for each key. For simplicity, we use Sequenti al SearchST methods. We need a cast when 
creating st[] because Java prohibits arrays with generics. The default constructor specifies 997 lists, 
so that for large tables, this code is about a factor of 1,000 faster than Sequenti al SearchST. This 
quick solution is an easy way to get good performance when you have some idea of the number of 
key-value pairs to be put() by a client. A more robust solution is to use array resizing to make sure 
that the lists are short no matter how many key-value pairs are in the table (see page 474 and exercise 
3 . 4 . 18 ). 
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Proposition K. In a separate-chaining hash table with M lists and N keys, the prob¬ 
ability (under assumption j) that the number of keys in a list is within a small 
constant factor of N/M is extremely close to 1. 


Proof sketch: assumption j makes this an application of classical probability 
theory. We sketch the proof, for readers who are familiar with basic probabilistic 
analysis. The probability that a given list will contain exactly k keys is given by the 
binomial distribution 



by the following argument: Choose k out of the N keys. Those k keys hash to the 
given list with probability 1/M, and the other N—k keys do not hash to the given 
list with probability 1 - (1/M). In terms of a — N/M, we can rewrite this expres¬ 
sion as . , / \k ( \N-k 

1 - 


a 

~N 


which (for small a) is closely 
approximated by the classical 
Poisson distribution 



k\ 
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Poisson distribution (N = 10 4 , M = 10 3 , a = 10) 


It follows that the probability that a list has more than t a keys on it is bounded 
by the quantity (a e/f) f e -a . This probability is extremely small for practical ranges 
of the parameters. For example, if the average length of the lists is 10, the prob¬ 
ability that we will hash to some list with more than 20 keys on it is less than (10 
e/2) 2 e -10 ~ 0.0084, and if the average length of the lists is 20, the probability that 
we will hash to some list with more than 40 keys on it is less than (20 el 2) 2 e -20 
0.0000016. This concentration result does not guarantee that every list will be 
short. Indeed it is known that, if a is a constant, the average length of the longest 
list grows with log N/ log log N. 
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This classical mathematical analysis is compelling, but it is important to note that 
it completely depends on assumption j. If the hash function is not uniform and inde¬ 
pendent, the search and insert cost could be proportional to N, no better than with 
sequential search, assumption j is much stronger than the corresponding assumption 
for other probabilistic algorithms that we have seen, and much more difficult to verify. 
With hashing, we are assuming that each and every key, no matter how complex, is 
equally likely to be hashed to one of M indices. We cannot afford to run experiments 
to test every possible key, so we would have to do more sophisticated experiments in¬ 
volving random sampling from the set of possible keys used in an application, followed 
by statistical analysis. Better still, we can use the algorithm itself as part of the test, to 
validate both assumption j and the mathematical results that we derive from it. 


Property L. In a separate-chaining hash table with M lists and N keys, the number 
of compares (equality tests) for search miss and insert is ~NIM. 

Evidence: Good performance of the algorithms in practice does not require the 
hash function to be fully uniform in the technical sense of assumption j. Count¬ 
less programmers since the 1950s have seen the speedups predicted by proposi¬ 
tion k, even for hash functions that are certainly not uniform. For example, the 
diagram on page 468 shows that list length distribution for our FrequencyCounter 
example (using our hash() implementation based on the hashCodeO from Java’s 
Stri ng data type) precisely matches the theoretical model. One exception that has 
been documented on numerous occasions is poor performance due to hash func¬ 
tions not taking all of the bits of the keys into account. Otherwise, the preponder¬ 
ance of the evidence from the experience of practical programmers puts us on 
solid ground in stating that hashing with separate chaining using an array of size M 
speeds up search and insert in a symbol table by a factor of M. 


Table size. In a separate-chaining implementation, our goal is to choose the table size 
M to be sufficiently small that we do not waste a huge area of contiguous memory 
with empty chains but sufficiently large that we do not waste time searching through 
long chains. One of the virtues of separate chaining is that this decision is not critical: 
if more keys arrive than expected, then searches will take a little longer than if we had 
chosen a bigger table size ahead of time; if fewer keys are in the table, then we have ex¬ 
tra-fast search with some wasted space. When space is not a critical resource, M can be 
chosen sufficiently large that search time is constant; when space is a critical resource, 
we still can get a factor of M improvement in performance by choosing M to be as 
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List lengths for java FrequencyCounter 8 < tale.txt using SeparateChainingHashST 

large as we can afford. For our example FrequencyCounter study, we see in the figure 
below a reduction in the average cost from thousands of compares per operation for 
Sequenti al SearchST to a small constant for SeparateChai ni ngHashST, as expected. 
Another option is to use array resizing to keep the lists short (see exercise 3 . 4 . 18 ). 

Deletion. To delete a key-value pair, simply hash to find the Sequential SearchST 
containing the key, then invoke the deleteO method for that table (see exercise 
3 . 1 . 5 ). Reusing code in this way is preferable to reimplementing this basic operation 
on linked lists. 

Ordered operations. The whole point of hashing is to uniformly disperse the keys, so 
any order in the keys is lost when hashing. If you need to quickly find the maximum 
or minimum key, find keys in a given range, or implement any of the other operations 
in the ordered symbol-table API on page 366, then hashing is not appropriate, since these 
operations will all take linear time. 

hashing with separate chaining is easy to implement and probably the fastest (and 
most widely used) symbol-table implementation for applications where key order is 
not important. When your keys are built-in Java types or your own type with well- 
tested implementations of hashCode(), algorithm 3.5 provides a quick and easy path 
to fast search and insert. Next, we consider an alternative scheme for collision resolu¬ 
tion that is also effective. 



Costsforjava FrequencyCounter 8 < tale.txt using SeparateChainingHashST (M = 997) 
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Hashing with linear probing Another approach to implementing hashing is to 
store N key-value pairs in a hash table of size M > N, relying on empty entries in the 
table to help with collision resolution. Such methods are called open-addressing hashing 
methods. 

The simplest open-addressing method is called linear probing: when there is a colli¬ 
sion (when we hash to a table index that is already occupied with a key different from 
the search key), then we just check the next entry in the table (by incrementing the 
index). Linear probing is characterized by identifying three possible outcomes: 

■ Key equal to search key: search hit 

■ Empty position (null key at indexed position): search miss 

■ Key not equal to search key: try next entry 

We hash the key to a table index, check whether the search key matches the key there, 
and continue (incrementing the index, wrapping back to the beginning of the table 
if we reach the end) until finding either the search key or an empty table entry. It is 
customary to refer to the operation of determining whether or not a given table entry 
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Trace of linear-probing ST implementation for standard indexing client 
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ALGORITHM 3.6 Hashing with linear probing 


public class LinearProbingHashST<Key, Value> 

{ 


private int N; 
private int M = 16; 
private Key[] keys; 
private Value[] vals; 


// number of key-value pairs in the table 
// size of linear-probing table 
// the keys 
// the values 


public Li nearProbingHashSTO 

{ 

keys = (Key[]) new 0bject[M]; 
vals = (Value[]) new 0bject[M]; 


private int hash(Key key) 

{ return (key.hashCodeO & 0x7fffffff) % M; } 

private void resizeO // See page 474. 

public void put(Key key, Value val) 

{ 

if (N >= M/2) resize(2*M); // double M (see text) 


i nt i ; 

for (i = hash(key); keys[i] != null; i = (i + 1) % M) 
if (keys[i].equals(key)) { vals[i] = val; return; } 
keys[i] = key; 
vals[i] = val; 

N++; 

} 

public Value get(Key key) 

{ 

for (int i = hash(key); keys[i] != null; i = (i + 1) % M) 
if (keys[i].equals(key)) 
return vals[i] ; 
return nul1 ; 

} 


This symbol-table implementation keeps keys and values in parallel arrays (as in Bi narySearchST) 
but uses empty spaces (marked by nul 1) to terminate clusters of keys. If a new key hashes to an empty 
entry, it is stored there; if not, we scan sequentially to find an empty position. To search for a key, we 
scan sequentially starting at its hash index until finding null (search miss) or the key (search hit). 
Implementation of keysQ is left as exercise 3 . 4 . 19 . 
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holds an item whose key is equal to the search key as a probe. We use the term inter¬ 
changeably with the term compare that we have been using, even though some probes 
are tests for null. 

The essential idea behind hashing with open addressing is this: rather than using mem¬ 
ory space for references in linked lists, we use it for the empty entries in the hash table, 
which mark the ends of probe sequences. As you can see from Li nearProbi ngHashST 
(algorithm 3 . 6 ), applying this idea to implement the symbol-table API is quite 
straightforward. We implement the table with parallel arrays, one for the keys and one 
for the values, and use the hash function as an index to access the data as just discussed. 

Deletion. How do we delete a key-value pair from a linear-probing table? If you think 
about the situation for a moment, you will see that setting the key’s table position to 
null will not work, because that might prematurely terminate the search for a key that 
was inserted into the table later. As an example, suppose that we try to delete C in this 
way in our trace example, then search for H. The 
hash value for H is 4, but it sits at the end of the 
cluster, in position 7. If we set position 5 to nul 1, 
then get() will not find H. As a consequence, we 
need to reinsert into the table all of the keys in the 
cluster to the right of the deleted key. This process 
is trickier than it might seem, so you are encour¬ 
aged to trace through the code at right for an ex¬ 
ample that exercises it (see exercise 3 . 4 . 17 ). 

as with separate chaining, the performance of 
hashing with open addressing depends on the ratio 

= N/M, but we interpret it differently. We refer 
to a as the load factor of a hash table. For separate 
chaining, a is the average number of keys per list 
and is generally larger than 1; for linear probing, 

is the percentage of table entries that are occu¬ 
pied; it cannot be greater than 1. In fact, we cannot 
let the load factor reach 1 (completely full table) 
in Li nearProbi ngHashST because a search miss 
would go into an infinite loop in a full table. In¬ 
deed, for the sake of good performance, we use array resizing to guarantee that the load 
factor is between one-eighth and one-half. This strategy is validated by mathematical 
analysis, which we consider before we discuss implementation details. 


public void delete(Key key) 

{ 

if (lcontains(key)) return; 
int i = hash(key); 
while (!key.equals(keys[i])) 
i = (i + 1) % M; 
keys [i ] = nul 1; 
val s [i ] = nul 1; 
i = (i + 1) % M; 
while (keys[i] != null) 

{ 

Key keyToRedo = keys[i]; 
Value valToRedo = val s[i ] ; 
keys [i ] = nul 1 ; 
val s [i ] = nul 1 ; 

N—; 

put(keyToRedo, valToRedo); 
i = (i + 1) % M; 

} 

N—; 

if (N > 0 N = M/8) resize(M/2); 


Deletion for linear probing 
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Clustering. The average cost of linear probing depends on the way in which the entries 
clump together into contiguous groups of occupied table entries, called clusters, when 

they are inserted. For example, when the key C is inserted 
in our example, the result is a cluster (A C S ) of length 
3, which means that four probes are needed to insert H 
because H hashes to the first position in the cluster. Short 
clusters are certainly a requirement for efficient perfor¬ 
mance. This requirement can be problematic as the table 
fills, because long clusters are common. Moreover, since 
all table positions are equally likely to be the hash value 
of the next key to be inserted (under the uniform hash¬ 
ing assumption), long clusters are more likely to increase 
in length than short ones, because a new key hashing to 
any entry in the cluster will cause the cluster to increase 
in length by 1 (and possibly much more, if there is just one table entry separating the 
cluster from the next one). Next, we turn to the challenge of quantifying the effect of 
clustering to predict performance in linear probing, and using that knowledge to set 
design parameters in our implementations. 
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Analysis of linear probing. Despite the relatively simple form of the results, precise 
analysis of linear probing is a very challenging task. Knuth’s derivation of the following 
formulas in 1962 was a landmark in the analysis of algorithms: 


Proposition M. In a linear-probing hash table with M lists and N = a M keys, the 
average number of probes (under assumption j) required is 


1 

2 



and 


1 

2 



for search hits and search misses (or inserts), respectively. In particular, when a 
is about 1/2, the average number of probes for a search hit is about 3/2 and for a 
search miss is about 5/2. These estimates lose a bit of precision as a approaches 1, 
but we do not need them for that case, because we will only use linear probing for 
less than one-half. 


Discussion: We compute the average by computing the cost of a search miss start¬ 
ing at each position in the table, then dividing the total by M. All search misses 
take at least 1 probe, so we count the number of probes after the first. Consider the 
following two extremes in a linear-probing table that is half full (M = 2 N): In the 
best case, table positions with even indices could be empty, and table positions with 
odd indices could be occupied. In the worst case, the first half of the table positions 
could be empty, and the second half occupied. The average length of the clusters 
in both cases is N/(2N) = 1/2, but the average number of probes for a search miss 
is 1 (all searches take at least 1 probe) plus (0 + 1 + 0 + 1 +. . . )/( 2 N) = 1/2 in the 
best case, and is 1 plus (N + (AT - 1) +...)/ (2 N) ~ N/A in the worst case. This 
argument generalizes to show that the average number of probes for a search miss 
is proportional to the squares of the lengths of the clusters: If a cluster is of length t, 
then the expression (t+ ( t — 1) + . . . + 2 + 1) / M = t(t + 1)/(2M) counts the con¬ 
tribution of that cluster to the grand total. The sum of the cluster lengths is N, so, 
adding this cost for all entries in the table, we find that the total average cost for a 
search miss is 1 + N/ ( 2 M) plus the sum of the squares of the lengths of the clusters, 
divided by 2M. Thus, given a table, we can quickly compute the average cost of a 
search miss in that table (see exercise 3 . 4 . 21 ). In general, the clusters are formed 
by a complicated dynamic process (the linear-probing algorithm) that is difficult to 
characterize analytically, and quite beyond the scope of this book. 
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proposition m tells us (under our usual assumption j) that we can expect a search to 
require a huge number of probes in a nearly full table (as a approaches 1 the values of 
the formulas describing the number of probes grow very large) but that the expected 
number of probes is between 1.5 and 2.5 if we can ensure that the load factor a is less 
than 1/2. Next, we consider the use of array resizing for this purpose. 

Array resizing We can use our standard array-resizing technique from chapter 
i to ensure that the load factor never exceeds one-half. First, we need a new construc¬ 
tor for Li nearProbi ngHashST that takes a fixed capacity as argument (add a line to 

the constructor in algorithm 
3.6 that sets M to the given value 
before creating the arrays). Next, 
we need the resizeO method 
given at left, which creates a new 
Li nearProbingHashST of the giv¬ 
en size, puts all the keys and values 
in the table in the new one, then 
rehashes all the keys into the new 
table. These additions allow us to 
implement array doubling. The call 
to resizeO in the first statement 
in put() ensures that the table is at 
most one-half full. This code builds a hash table twice the size with the same keys, thus 
halving the value of a. As in other applications of array resizing, we also need to add 

if (N > 0 && N <= M/8) resize(M/2); 

as the last statement in deleteO to ensure that the table is at least one-eighth full. 
This ensures that the amount of memory used is always within a constant factor of the 
number of key-value pairs in the table. With array resizing, we are assured that a < 1/2. 

Separate chaining. The same method works to keep lists short (of average 
length between 2 and 8 ) in separate chaining: replace Li nearProbi ngHashST by 
SeparateChai ni ngHashST in resi ze(), call resi ze(2*M) when (N >= M/2) in put(), 
and call resi ze(M/2) when (N > 0 && N <= M/8) in del ete(). For separate chain¬ 
ing, array resizing is optional and not worth your trouble if you have a decent estimate 
of the client’s N: just pick a table size M based on the knowledge that search times are 
proportional to 1+ N/M. For linear probing, array resizing is necessary. A client that 
inserts more key-value pairs than you expect will encounter not just excessively long 
search times, but an infinite loop when the table fills. 


private void resize(int cap) 

{ 

LinearProbingHashST<Key, Values t; 
t = new LinearProbingHashST<Key, Value>(cap); 
for (int i =0; i < M; i++) 
if (keys[i] != null) 

t. put(keys [i ] , vals[i]); 
keys = t.keys; 
vals = t.vals; 

M = t.M; 

} 

Resizing a linear-probing hash table 
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Amortized analysis. From a theoretical standpoint, when we use array resizing, we 
must settle for an amortized bound, since we know that those insertions that cause the 
table to double will require a large number of probes. 

Proposition N. Suppose a hash table is built with array resizing, starting with 
an empty table. Under assumption j, any sequence of t search, insert, and delete 
symbol-table operations is executed in expected time proportional to t and with 
memory usage always within a constant factor of the number of keys in the table. 

Proof.: For both separate chaining and linear probing, this fact follows from a sim¬ 
ple restatement of the amortized analysis for array growth that we first discussed in 
chapter 1, coupled with proposition k and proposition m. 



Costsfor java FrequencyCounter 8 < tale.txt using SeparateChainingHashST (with doubling) 



Costs for java FrequencyCounter 8 < tale.txt using Li nearProbingHashST (with doubling) 
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The plots of the cumulative averages for our FrequencyCounter example (shown at 
the bottom of the previous page) nicely illustrate the dynamic behavior of array resiz¬ 
ing in hashing. Each time the array doubles, the cumulative average increases by about 
1 , because each key in the table needs to be rehashed; then it decreases because about 
half as many keys hash to each table position, with the rate of decrease slowing as the 
table fills again. 

Memory As we have indicated, understanding memory usage is an important factor 
if we want to tune hashing algorithms for optimum performance. While such tuning 
is for experts, it is a worthwhile exercise to calculate a rough estimate of the amount of 
memory required, by estimating the number of references used, as follows: Not counting 
the memory for keys and values, our implementation SeparateChai ni ngHashST uses 
memoryforMreferencesto Sequent! ai SearchST objectsplusMSequenti ai SearchST 
objects. Each Sequent! al SearchST object has the usual 16 bytes of object overhead 
plus one 8 -byte reference (f i rst), and there are a total of N Node objects, each with 24 
bytes of object overhead plus 3 references (key, val ue, and next). This compares with 
an extra reference per node for binary search trees. With array resizing to ensure that 
the table is between one-eighth and one-half full, linear probing uses between 4 N and 
16/V references. Thus, choosing hashing on the basis of memory usage is not normally 
justified. The calculation is a bit different for primitive types (see exercise 3 . 4 . 24 ) 


method 


space usage for N items 
(reference types) 


separate chaining ~ 48 N + 64 M 


. . between 

linear probing _ 32Nand _ 128N 


BSTs 


5 6N 


Space usage in symbol tables 
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since the earliest days of computing, researchers have studied (and are study¬ 
ing) hashing and have found many ways to improve the basic algorithms that we have 
discussed. You can find a huge literature on the subject. Most of the improvements 
push down the space-time curve: you can get the same running time for searches using 
less space or get faster searches using the same amount of space. Other improvements 
involve better guarantees, on the expected worst-case cost of a search. Others involve 
improved hash-function designs. Some of these methods are addressed in the exercises. 

Detailed comparison of separate chaining and linear probing depends on myriad 
implementation details and on client space and time requirements. It is not normally 
justified to choose separate chaining over linear probing on the basis of performance 
(see exercise 3 . 5 . 31 ). In practice, the primary performance difference between the two 
methods has to do with the fact that separate chaining uses a small block of memory 
for each key-value pair, while linear probing uses two large arrays for the whole table. 
For huge tables, these needs place quite different burdens on the memory management 
system. In modern systems, this sort of tradeoff is best addressed by experts in extreme 
performance-critical situations. 

With hashing, under generous assumptions, it is not unreasonable to expect to 
support the search and insert symbol-table operations in constant time, independent 
of the size of the table. This expectation is the theoretical optimum performance for 
any symbol-table implementation. Still, hashing is not a panacea, for several reasons, 
including: 

■ A good hash function for each type of key is required. 

■ The performance guarantee depends on the quality of the hash function. 

■ Hash functions can be difficult and expensive to compute. 

■ Ordered symbol-table operations are not easily supported. 

Beyond these basic considerations, we defer the comparison of hashing with the other 
symbol-table methods that we have studied to the beginning of section 3 . 5 . 
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| Q&A 


Q. How does Java implement hashCodeO for Integer, Doubl e, and Long? 

A. For Integer it just returns the 32-bit value. For Double and Long it returns the ex¬ 
clusive or of the first 32 bits with the second 32 bits of the standard machine representa¬ 
tion of the number. These choices may not seem to be very 
random, but they do serve the purpose of spreading out the 
values. 

Q. When using array resizing, the size of the table is always 
a power of 2. Isn’t that a potential problem, because it only 
uses the least significant bits of hashCodeO? 

A. Yes, particularly with the default implementations. One 
way to address this problem is to first distribute the key val¬ 
ues using a prime larger than M, as in the following example: 

private int hash(Key x) 

{ 

int t = x.hashCodeO & 0x7fffffff; 
if (lgM < 26) t = t % primes[lgM+5] ; 
return t % M; 

} 

This code assumes that we maintain an instance variable 
1 gM that is equal to lg M (by initializing to the appropri¬ 
ate value, incrementing when doubling, and decrementing 
when halving) and an array pri mes [] of the smallest prime 
greater than each power of 2 (see the table at right). The 
constant 5 is an arbitrary choice—we expect the first % to 
distribute the values equally among the values less than the 
prime and the second to map about five of those values to 
each value less than M. Note that the point is moot for large 
M. 


k 

5* 

primes[k] 

(2 fc -S fc ) 

5 

1 

31 

6 

3 

61 

7 

1 

127 

8 

5 

251 

9 

3 

509 

10 

3 

1021 

11 

9 

2039 

12 

3 

4093 

13 

1 

8191 

14 

3 

16381 

15 

19 

32749 

16 

15 

65521 

17 

1 

131071 

18 

5 

262139 

19 

1 

524287 

20 

3 

1048573 

21 

9 

2097143 

22 

3 

4194301 

23 

15 

8388593 

24 

3 

16777213 

25 

39 

33554393 

26 

5 

67108859 

27 

39 

134217689 

28 

57 

268435399 

29 

3 

536870909 

30 

35 

1073741789 

31 

1 

2147483647 


Primes for hash table sizes 


Q. I’ve forgotten. Why don’t we implement hash (x) by returning x. hashCodeO % M? 
A. We need a result between 0 and M-l, but in Java, the % function may be negative. 

Q. So, why not implement hash (x) by returning Math. abs(x. hashcodeO) % M? 
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A. Nice try. Unfortunately, Math. abs () returns a negative result for the largest nega¬ 
tive number. For many typical calculations, this overflow presents no real problem, 
but for hashing it would leave you with a program that is likely to crash after a few bil¬ 
lion inserts, an unsettling possibility. For example, s. hashCode () is — 2 31 for the Java 
String value "polygene!ubricants". Finding other strings that hash to this value 
(and to 0 ) has turned into an amusing algorithm-puzzle pastime. 

Q. Why not use Bi narySearchST or RedBl ackBST instead of Seguenti al SearchST in 
ALGORITHM 3 . 5 ? 

A. Generally, we set parameters so as to make the number of keys hashing to each value 
small, and elementary symbol tables are generally better for the small tables. In certain 
situations, slight performance gains may be achieved with such hybrid methods, but 
such tuning is best left for experts. 

Q. Is hashing faster than searching in red-black BSTs? 

A. It depends on the type of the key, which determines the cost of computing 
hashCode () versus the cost of compareToC). For typical key types and for Java default 
implementations, these costs are similar, so hashing will be significantly faster, since it 
uses only a constant number of operations. But it is important to remember that this 
question is moot if you need ordered operations, which are not efficiently supported in 
hash tables. See section 3.5 for further discussion. 

Q. Why not let the linear probing table get, say, three-quarters full? 

A. No particular reason. You can choose any value of a, using proposition m to esti¬ 
mate search costs. For a = 3/4, the average cost of search hits is 2.5 and search misses is 
8.5, but if you let a grow to 7/8, the average cost of a search miss is 32.5, perhaps more 
than you want to pay. As a gets close to 1, the estimate in proposition m becomes in¬ 
valid, but you don’t want your table to get that close to being full. 
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EXERCISES 


3 . 4.1 Insert the keys EASYQUTIONin that order into an initially empty table 
of M = 5 lists, using separate chaining. Use the hash function 11 k % M to transform 
the /cth letter of the alphabet into a table index. 

3 . 4.2 Develop an alternate implementation of SeparateChai ni ngHashST that directly 
uses the linked-list code from Sequenti al SearchST. 

3 . 4.3 Modify your implementation of the previous exercise to include an integer field 
for each key-value pair that is set to the number of entries in the table at the time that 
pair is inserted. Then implement a method that deletes all keys (and associated values) 
for which the field is greater than a given integer k. Note: This extra functionality is use¬ 
ful in implementing the symbol table for a compiler. 

3 . 4.4 Write a program to find values of a and M, with M as small as possible, such that 
the hash function (a * k) % M for transforming the /cth letter of the alphabet into a 
table index produces distinct values (no collisions) for the keys SEARCHXMPL. 
The result is known as a perfect hash function. 

3 . 4.5 Is the following implementation of hashCode() legal? 

public int hashCodeO 
{ return 17; } 

If so, describe the effect of using it. If not, explain why. 

3 . 4.6 Suppose that keys are f-bit integers. For a modular hash function with prime M, 
prove that each key bit has the property that there exist two keys differing only in that 
bit that have different hash values. 

3 . 4.7 Consider the idea of implementing modular hashing for integer keys with the 
code (a * k) % M , where a is an arbitrary fixed prime. Does this change mix up the 
bits sufficiently well that you can use nonprime M? 

3 . 4.8 How many empty lists do you expect to see when you insert N keys into a hash 
table with SeparateChai ni ngHashST, for N=10, 10 2 , 10 3 , 10 4 , 10 5 , and 10 6 ? Hint: See 
exercise 2 . 5 . 31 . 

3 . 4.9 Implement an eager del ete() method for SeparateChai ni ngHashST. 

3 . 4.10 Insert the keys EASYQUTIONin that order into an initially empty table 
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of size M =16 using linear probing. Use the hash function 11 k % M to transform the 
kth letter of the alphabet into a table index. Redo this exercise for M = 10. 

3 . 4 . 11 Give the contents of a linear-probing hash table that results when you insert the 
keys EASYQUTIONin that order into an initially empty table of initial size M 
= 4 that is expanded with doubling whenever half full. Use the hash function 11 k % M 
to transform the /cth letter of the alphabet into a table index. 


3 . 4.12 Suppose that the keys A through C, with the hash values given below, are inserted 
in some order into an initially empty table of size 7 using a linear-probing table (with 
no resizing for this problem). Which of the following could not possibly result from 
inserting these keys? 


a. E F G 

b. CEB 

c. B D F 

d. C C B 

e. F C B 

/ C E C 


A C 
C F 
A C 
A D 
D A 
A D 


B 

D 

E 

E 

C 

B 


D 

A 

G 

F 

E 

F 


Give the minimum and the maximum number of probes that could be required to 
build a table of size 7 with these keys, and an insertion order that justifies your answer. 


3 . 4.13 Which of the following scenarios leads to expected linear running time for a 
random search hit in a linear-probing hash table? 


a. All keys hash to the same index. 

b. All keys hash to different indices. 

c. All keys hash to an even-numbered index. 

d. All keys hash to different even-numbered indices. 


3 . 4.14 Answer the previous question for search miss, assuming the search key is equally 
likely to hash to each table position. 

3 . 4.15 How many compares could it take, in the worst case, to insert N keys into an 
initially empty table, using linear probing with array resizing? 

3 . 4.16 Suppose that a linear-probing table of size 10 6 is half full, with occupied posi¬ 
tions chosen at random. Estimate the probability that all positions with indices divisible 
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EXERCISES (continued) 


by 100 are occupied. 

3 . 4.17 Show the result of using the del ete () method on page 471 to delete C from the 
table resulting from using Li nearProbi ngHashST with our standard indexing client 
(shown on page 469). 

3 . 4.18 Add a constructor to SeparateChai ni ngHashST that gives the client the ability 
to specify the average number of probes to be tolerated for searches. Use array resizing 
to keep the average list size less than the specified value, and use the technique described 
on page 478 to ensure that the modulus for hash () is prime. 

3 . 4.19 Implement keys () for SeparateChai ni ngHashST and Li nearProbi ngHashST. 

3 . 4.20 Add a method to Li nearProbi ngHashST that computes the average cost of a 
search hit in the table, assuming that each key in the table is equally likely to be sought. 

3 . 4.21 Add a method to Li nearProbi ngHashST that computes the average cost of a 
search miss in the table, assuming a random hash function. Note'. You do not have to 
compute any hash functions to solve this problem. 

3 . 4.22 Implement hashCodeO for various types: Point2D, Interval, Interval 2D, 
and Date. 

3 . 4.23 Consider modular hashing for string keys with R = 256 and M = 255. Show 
that this is a bad choice because any permutation of letters within a string hashes to the 
same value. 

3 . 4.24 Analyze the space usage of separate chaining, linear probing, and BSTs for 
doubl e keys. Present your results in a table like the one on page 476. 
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CREATIVE PROBLEMS 


3 . 4.25 Hash cache. Modify T ransacti on on page 462 to maintain an instance variable 
hash, so that hashCodeO can save the hash value the first time it is called for each object 
and does not have to recompute it on subsequent calls. Note: This idea works only for 
immutable types. 

3 . 4.26 Lazy delete for linear probing. Add to LinearProbingHashST a deleteO 
method that deletes a key-value pair by setting the value to null (but not removing 
the key) and later removing the pair from the table in resize (). Your primary chal¬ 
lenge is to decide when to call resi ze(). Note: You should overwrite the nul 1 value if 
a subsequent put() operation associates a new value with the key. Make sure that your 
program takes into account the number of such tombstone items, as well as the number 
of empty positions, in making the decision whether to expand or contract the table. 

3 . 4.27 Double probing. Modify SeparateChai ni ngHashST to use a second hash func¬ 
tion and pick the shorter of the two lists. Give a trace of the process of inserting the keys 
EASYQUTIONin that order into an initially empty table of size M =3 using 
the function 11 k % M (for the fcth letter) as the first hash function and the function 
17 k % M (for the kth letter) as the second hash function. Give the average number of 
probes for random search hit and search miss in this table. 

3 . 4.28 Double hashing. Modify Li nearProbi ngHashST to use a second hash function 
to define the probe sequence. Specifically, replace (i + 1) % M (both occurrences) by 
(i + k) % M where k is a nonzero key-dependent integer that is relatively prime to M. 
Note: You may meet the last condition by assuming that M is prime. Give a trace of the 
process of inserting the keys EASYQUTIONin that order into an initially empty 
table of size M =11, using the hash functions described in the previous exercise. Give 
the average number of probes for random search hit and search miss in this table. 

3 . 4.29 Deletion. Implement an eager del ete () method for the methods described in 
each of the previous two exercises. 

3 . 4.30 Chi-square statistic. Add a method to SeparateChai ni ngST to compute the X 2 
statistic for the hash table. With N keys and table size M, this number is defined by the 
equation 

X 2 = (M/N) ( (/ 0 - N/M) 2 + (/j - N/M) 2 + ... N/M) 2 ) 
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CREATIVE PROBLEMS (continued) 


where/; is the number of keys with hash value i. This statistic is one way of checking our 
assumption that the hash function produces random values. If so, this statistic, for N > 
cM, should be between M — \ M and M + f M with probability 1 — 1/c. 

3 . 4.31 Cuckoo hashing. Develop a symbol-table implementation that maintains two 
hash tables and two hash functions. Any given key is in one of the tables, but not both. 
When inserting a new key, hash to one of the tables; if the table position is occupied, 
replace that key with the new key and hash the old key into the other table (again kick¬ 
ing out a key that might reside there). If this process cycles, restart. Keep the tables less 
than half full. This method uses a constant number of equality tests in the worst case 
for search (trivial) and amortized constant time for insert. 

3 . 4.32 Hash attack. Find 2 N strings, each of length 2 N , that have the same hashCode O 
value, supposing that the hashCodeO implementation for Stri ng is the following: 

public int hashCodeO 

{ 

int hash = 0; 

for (int i =0; i < lengthO; i ++) 
hash = (hash * Bl) + charAt(i); 
return hash; 

} 

Strong hint'. Aa and BB have the same value. 

3 . 4.33 Bad hash function. Consider the following hashCodeO implementation for 
Stri ng, which was used in early versions of Java: 

public int hashCodeO 

{ 

int hash = 0; 

int skip = Math.max(l, length()/8); 
for (int i =0; i < lengthO; i += skip) 
hash = (hash * B7) + charAt(i); 
return hash; 

} 

Explain why you think the designers chose this implementation and then why you 
think it was abandoned in favor of the one in the previous exercise. 
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3 . 4.34 Hash cost. Determine empirically the ratio of the time required for hash() 
to the time required for compareToO, for as many commonly-used types of keys for 
which you can get meaningful results. 

3 . 4.35 Chi-square test. Use your solution from exercise 3 . 4.30 to check the assump¬ 
tion that the hash functions for commonly-used key types produce random values. 

3 . 4.36 List length range. Write a program that inserts N random i nt keys into a table 
of size N /100 using separate chaining, then finds the length of the shortest and longest 
lists, for N = 10 3 , 10 4 , 10 5 , 10 6 . 

3 . 4.37 Hybrid. Run experimental studies to determine the effect of using RedBl ackBST 
instead of SequentialSearchST to handle collisions in SeparateChainingHashST. 
This solution carries the advantage of guaranteeing logarithmic performance even for 
a bad hash function and the disadvantage of necessitating maintenance of two different 
symbol-table implementations. What are the practical effects? 

3 . 4.38 Separate-chaining distribution. Write a program that inserts 10 5 random non¬ 
negative integers less than 10 6 into a table of size 10 5 using linear probing, and that 
plots the total number of probes used for each 10 3 consecutive insertions. Discuss the 
extent to which your results validate proposition k. 

3 . 4.39 Linear-probing distribution. Write a program that inserts N/2 random i nt keys 
into a table of size N using linear probing, then computes the average cost of a search 
miss in the resulting table from the cluster lengths, for N= 10 3 , 10 4 , 10 5 , 10 6 . Discuss 
the extent to which your results validate proposition m. 

3 . 4.40 Plots. Instrument Li nearProbi ngHashST and SeparateChai ni ngHashST to 
produce plots like the ones shown in the text. 

3 . 4.41 Double probing. Run experimental studies to evaluate the effectiveness of dou¬ 
ble probing (see exercise 3 . 4 . 27 ). 

3 . 4.42 Double hashing. Run experimental studies to evaluate the effectiveness of dou¬ 
ble hashing (see exercise 3 . 4 . 28 ). 

3 . 4.43 Parking problem. (D. Knuth) Run experimental studies to validate the hypoth¬ 
esis that the number of compares needed to insert M random keys into a linear-probing 
table of size M is ~cM 3/2 , where c = x tt/2- 
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from the early days of computing, when symbol tables allowed programmers to 
progress from using numeric addresses in machine language to using symbolic names 
in assembly language, to modern applications of the new millennium, when symbolic 
names have meaning across worldwide computer networks, fast search algorithms have 
played and continue to play an essential role in computation. Modern applications for 
symbol tables include organization of scientific data, from searching for markers or 
patterns in genomic data to mapping the universe; organization of knowledge on the 
web, from searching in online commerce to putting libraries online; and implement¬ 
ing the internet infrastructure, from routing packets among machines on the web to 
shared file systems and video streaming. Efficient search algorithms have enabled these 
and countless other important applications. We will consider several representative ex¬ 
amples in this section: 

■ A dictionary client and an indexing client that enable fast and flexible access to 
information in comma-separated-value files (and similar formats), which are 
widely used to store data on the web 

■ An indexing client for building an inverted index of a set of files 

■ A sparse-matrix data type that uses a symbol table to address problem sizes far 
beyond what is possible with the standard implementation 

In chapter 6 , we consider a symbol table that is appropriate for tables such as data¬ 
bases and file systems that contain a vast number of keys, as large as can be reasonably 
contemplated. 

Symbol tables also play a critical role in algorithms that we consider throughout the 
rest of the book. For example, we use symbol tables to represent graphs (chapter 4 ) 
and to process strings (chapter 5 ). 

As we have seen throughout this chapter, developing symbol-table implementations 
that can guarantee fast performance for all operations is certainly a challenging task. 
On the other hand, the implementations that we have considered are well-studied, 
widely used, and available in many software environments (including Java libraries). 
From this point forward, you certainly should consider the symbol-table abstraction to 
be a key component in your programmer’s toolbox. 
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Which symbol-table implementation should I use? The table at the bottom 
of this page summarizes the performance characteristics of the algorithms that we have 
considered in propositions and properties in this chapter (with the exception of the 
worst-case results for hashing, which are from the research literature and unlikely to 
be experienced in practice). It is clear from the table that, for typical applications, your 
decision comes down to a choice between hash tables and binary search trees. 

The advantages of hashing over BST implementations are that the code is simpler 
and search times are optimal (constant), if the keys are of a standard type or are suf¬ 
ficiently simple that we can be confident of developing an efficient hash function for 
them that (approximately) satisfies the uniform hashing assumption. The advantages 
of BSTs over hashing are that they are based on a simpler abstract interface (no hash 
function need be designed); red-black BSTs can provide guaranteed worst-case perfor¬ 
mance; and they support a wider range of operations (such as rank, select, sort, and 
range search). As a rule of thumb, most programmers will use hashing except when 
one or more of these factors is important, when red-black BSTs are called for. In chap¬ 
ter 5 , we will study one exception to this rule of thumb: when keys are long strings, 
we can build data structures that are even more flexible than red-black BSTs and even 
faster than hashing. 


algorithm 
(data structure) 

worst-case cost 
(after N inserts) 

search insert 

average-case cost 
(after N random inserts) 

search hit insert 

key 

interface 

memory 

(bytes) 

sequential search 
(unordered list) 

N 

N 

NI2 

N 

equalsO 

48 N 

binary search 
(ordered array) 

lg N 

N 

lg N 

N/2 

compareToC) 

16 N 

binary tree search 
(BST) 

N 

N 

1.39 lg N 

1.39 lg N 

compareToC) 

64 N 

2-3 tree search 
(red-black BST) 

2 lg N 

2 lg N 

1.00 IgN 

1.00 lgiV 

compareToC) 

64 N 

separate chainingt 
(array of lists) 

<lgN 

<lg N 

N/(2M) 

N/M 

equals() 

hashCodeO 

48 N + 64 M 

linear probing^ 
(parallel arrays) 

clgN 

c\gN 

< 1.50 

<2.50 

equal s() 
hashCodeO 

between 

32 N and 128 iV 


f with uniform and independent hash function 


Asymptotic cost summary for symbol-table implementations 
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Our symbol-table implementations are useful for a wide range of applications, but 
our algorithms are easily adapted to support several other options that are widely used 
and worth considering. 

Primitive types. Suppose that we have a symbol table with integer keys and associ¬ 
ated floating-point numbers. When we use our standard setup, the keys and values 
are stored as Integer and Doubl e wrapper-type values, so we need two extra memory 
references to access each key-value pair. These references may be no problem in an ap¬ 
plication that involves thousands of searches on thousands of keys but may represent 
excessive cost in an application that involves billions of searches on millions of keys. Us¬ 
ing a primitive type instead of Key would save one reference per key-value pair. When 
the associated value is also primitive, we can eliminate another reference. The situation 
is diagrammed at right for separate chaining; the 
same tradeoffs hold for other implementations. For 
performance-critical applications, it is worthwhile 
and not difficult to develop versions of our imple¬ 
mentations along these lines (see exercise 3.5.4). 

Duplicate keys. The possibility of duplicate keys 
sometimes needs special consideration in symbol- 
table implementations. In many applications, it is 
desirable to associate multiple values with the same 
key. For example, in a transaction-processing sys¬ 
tem, numerous transactions may have the same 
customer key value. Our convention to disallow 
duplicate keys amounts to leaving duplicate-key 
management to the client. We will consider an ex¬ 
ample of such a client later in this section. In many 
of our implementations, we could consider the al¬ 
ternative of leaving key-value pairs with duplicate 
keys in the primary search data structure and to return any value with the given key for 
a search. We might also add methods to return all values with the given key. Our BST 
and hashing implementations are not difficult to adapt to keep duplicate keys within 
the data structure; doing so for red-black BSTs is just slightly more challenging (see ex¬ 
ercise 3.5.9 and exercise 3.5.10). Such implementations are common in the literature 
(including earlier editions of this book). 


standard implementation 


data is stored in 



primitive-type implementation 


data is stored in 
linked-list nodes 


O-tP-E 


Memory usage for separate chaining 
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Java libraries. Java’s java, uti 1 .TreeMap and java, uti 1 . HashMap libraries are sym¬ 
bol-table implementations based on red-black BSTs and hashing with separate chaining 
respectively. TreeMap does not directly support rank(), selectO, and other opera¬ 
tions in our ordered symbol-table API, but it does support operations that enable effi¬ 
cient implementation of these. HashMap is roughly equivalent to our Li nearProbi ngST 
implementation—it uses array resizing to enforce a load factor of about 75 percent. 

to be consistent and specific, we use in this book the symbol-table implementation 
based on red-black BSTs from section 3.3 or the one based on linear-probing hashing 
from section 3.4. For economy and to emphasize client independence from specific 
implementations, we use the name ST as shorthand for RedBl ackBST for ordered sym¬ 
bol tables in client code and the name HashST as shorthand for Li nearProbi ngHashST 
when order is not important and hash functions are available. We adopt these conven¬ 
tions with full knowledge that specific applications might have demands that could call 
for some variation or extension of one of these algorithms and data structures. Which 
symbol table should you use? Whatever you decide, test your choice to be sure that it is 
delivering the performance that you expect. 

Set APIs Some symbol-table clients do not need the values, just the ability to insert 
keys into a table and to test whether a key is in the table. Because we disallow duplicate 
keys, these operations correspond to the following API where we are just interested in 
the set of keys in the table, not any associated values: 


public class SET<Key> 



SET 0 

create an empty set 

voi d 

add(Key key) 

add key into the set 

voi d 

delete(Key key) 

remove key from the set 

boolean 

contains(Key key) 

is key in the set? 

boolean 

i sEmptyO 

is the set empty? 

int 

size() 

number of keys in the set 

String 

toString() 

string representation of the set 


API for a basic set data type 


You can turn any symbol-table implementation into a SET implementation by ignoring 
values or by using a simple wrapper class (see exercises 3.5.1 through 3 . 5 . 3 ). 
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Extending SET to include union, intersection, complement, and other common math¬ 
ematical set operations requires a more sophisticated API (for example, the comple¬ 
ment operation requires some mechanism for specifying a universe of all possible keys) 
and provides a number of interesting algorithmic challenges, as discussed in exercise 
3 - 5 - 17 - 

As with ST, we have unordered and ordered versions of SET. If keys are Comparabl e, 
we can include min(), maxO, floorC), ceilingO, deleteMinO, deleteMaxO, 
rank(), selectO, and the two-argument versions of size() and get() to define a 
full API for ordered keys. To match our ST conventions, we use the name SET in client 
code for ordered sets and the name Hash SET when order is not important. 

To illustrate uses of SET, we consider filter clients that read a sequence of strings 
from standard input and write some of them to standard output. Such clients have 
their origin in early systems where main memory was far too small to hold all the data, 
and they are still relevant today, when we write programs that take their input from the 
web. As example input, we use ti nyTal e. txt (see page 371 ). For readability, we preserve 
newlines from the input to the 
output in examples, even though 
the code does not do so. 

Dedup. The prototypical filter 
example is a SET or HashSET cli¬ 
ent that removes duplicates in 
the input stream. It is custom¬ 
ary to refer to this operation as 
dedup. We maintain a set of the 
string keys seen so far. If the next 
key is in the set, ignore it; if it is 
not in the set, add it to the set 
and print it. The keys appear on 
standard output in the order they 

appear on standard input, with , 
rr i , Dedup filter 

duplicates removed. This process 

takes space proportional to the 

number of distinct keys in the 

input stream (which is typically 

far smaller than the total number 

of keys). 


% java DeDup < tinyTale.txt 
it was the best of times worst 
age wisdom foolishness 
epoch belief incredulity 
season light darkness 
spring hope winter despair 


public class DeDup 
{ 

public static void main(String[] args) 
{ 

HashSET<String> set; 

set = new HashSET<String>(); 

while ( ! Stdln . i sEmptyO) 

{ 

String key = Stdln. readStri ng() ; 
if (!set.contains(key)) 

{ 

set.add(key); 

StdOut.println(key); 

} 

} 

} 
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Whitelist and blacklist. Another classic filter uses keys in a separate file to decide 
which keys from the input stream are passed to the output stream. This general process 
has many natural applications. The simplest example is a whitelist, where any key that 
is in the file is identified as “good.” The client might choose to pass through to standard 
output any key that is not in the whitelist and to ignore any key that is in the whitelist 
(as in the example considered in our first program in chapter i); another client might 
choose to pass through to standard output any key that is in the whitelist and to ig¬ 
nore any key that is not in the whitelist (as 
shown in the HashSET client Whi teFi 1 ter 
at right). For example, your email applica¬ 
tion might use such a filter to allow you to 
specify the addresses of your friends and 
to direct it to consider emails from any¬ 
one else as spam. We build a HashSET of 
the keys in the specified list, then read the 
keys from standard input. If the next key 
is in the set, print it; if it is not in the set, 
ignore it. A blacklist is the opposite, where 
any key that is in the file is identified as 
“bad.” Again, there are two natural filters 
for clients using a blacklist. In our email 
example, you might specify the addresses 
of known spammers and direct the email whitelist filter 
application to let through all mail not 
from one of those addresses. We can im¬ 
plement a HashSET client BlackFilter 
that implements this filter by negating the 
filter test in Whi teFi Iter. Typical practi¬ 
cal situations such as a credit card com¬ 
pany using a blacklist to filter out stolen 
card numbers or an internet router using a 
whitelist to implement a firewall are likely 
to involve huge lists, unbounded input 
streams, and strict response requirements. 

The sorts of symbol-table implementa¬ 
tions that we have considered enable such 
challenges to easily be met. 


% more list.txt 
was it the of 

% java Whi teFi Iter list.txt < tinyTale.txt 

it was the of it was the of 

it was the of it was the of 

it was the of it was the of 

it was the of it was the of 

it was the of it was the of 

% java BlackFilter list.txt < tinyTale.txt 

best times worst times 

age wisdom age foolishness 

epoch belief epoch incredulity 

season light season darkness 

spring hope winter despair 


public class WhiteFilter 
{ 

public static void main(String[] args) 

{ 

HashSET<String> set; 

set = new HashSET<String>(); 

In in = new ln(args[0]); 
while (! i n . i sEmptyO) 

set.add(in. readStringO) ; 
while (! Stdln . i sEmptyO) 

{ 

String word = Stdln . readStri ng() ; 
if (set. contai ns(word)) 

StdOut.println(word); 

} 

} 
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Dictionary clients The most basic kind of symbol-table client builds a symbol 
table with successive put operations in order to support get requests. Many applications 
also take advantage of the idea that a symbol table is a dynamic dictionary, where it is 
easy to look up information and to update the information in the table. The following 
list of familiar examples illustrates the utility of this approach: 

■ Phone book. When keys are people’s names and values are their phone num¬ 
bers, a symbol table models a phone book. A very significant difference from 
a printed phone book is that we can add new names or change existing phone 
numbers. We could also use the phone number as the key and the name as the 
value—if you have never done so, try typing your phone number (with area 
code) into the search field in your browser. 

■ Dictionary. Associating a word with its definition is a familiar concept that 
gives us the name “dictionary.” For centuries people kept printed dictionaries in 
their homes and offices in order to check the definitions and spellings (values) 
of words (keys). Now, because of good symbol-table implementations, people 
expect built-in spell checkers and immediate access to word definitions on their 
computers. 

■ Account information. People who own stock now regularly check the current 
price on the web. Several services on the web associate a ticker symbol (key) with 
the current price (value), usually along with a great deal of other information. 
Commercial applications of this sort abound, including financial institutions 
associating account information with a name or account number or educational 
institutions associating grades with a student name or identification number. 

■ Genomics. Symbols play a central role in modern genomics. The simplest ex¬ 
ample is the use of the letters A, C, T, and C to represent the nucleotides found in 
the DNA of living organisms. The next simplest is the correspondence between 
codons (nucleotide triplets) and amino acids (TTA corresponds to leucine, TCT 
to serine, and so forth), then the correspondence between sequences of amino 
acids and proteins, and so forth. Researchers in genomics routinely use various 
types of symbol tables to organize this knowledge. 

■ Experimental data. From astrophysics to zoology, modern scientists are awash in 
experimental data, and organizing and efficiently accessing this data are vital to 
understanding what it means. Symbol tables are a critical starting point, and ad¬ 
vanced data structures and algorithms that are based on symbol tables are now 
an important part of scientific research. 

■ Compilers. One of the earliest uses of symbol tables was to organize information 
for programming. At first, programs were simply sequences of numbers, but 
programmers very quickly found that using symbolic names for operations and 
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memory locations (variable names) was far more convenient. Associating the 
names with the numbers requires a symbol table. As the size of programs grew, 
the cost of the symbol-table operations became a bottleneck in program devel¬ 
opment time, which led to the development of data structures and algorithms 
like the ones we consider in this chapter. 

■ File systems. We use symbol tables regularly to 
organize data on computer systems. Perhaps the 
most prominent example is the file system, where 
we associate a file name (key) with the location 
of its contents (value). Your music player uses the 
same system to associate song titles (keys) with 
the location of the music itself (value). 

■ Internet DNS. The domain name system (DNS) 
that is the basis for organizing information on 
the internet associates URLs (keys) that humans 
understand (such as www. pri nceton.edu or 
www .wikipedia.org) with IP addresses (values) 
that computer network routers understand (such 
as 208.216.181.15 or 207.142.131.206). This 
system is the next-generation “phone book.” 

Thus, humans can use names that are easy to re¬ 
member and machines can efficiently process the 
numbers. The number of symbol-table lookups 
done each second for this purpose on internet 
routers around the world is huge, so perfor¬ 
mance is of obvious importance. Millions of new computers and other devices 
are put onto the internet each year, so these symbol tables on internet routers 
need to be dynamic. 

Despite its scope, this list is still just a representative sample, intended to give you a fla¬ 
vor of the scope of applicability of the symbol-table abstraction. Whenever you specify 
something by name, there is a symbol table at work. Your computer’s file system or the 
web might do the work for you, but there is still a symbol table there somewhere. 

As a specific example, we consider a symbol-table client that you can use to look up 
information that is kept in a table on a file or a web page using the comma-separated- 
value (. csv) file format. This simple format achieves the (admittedly modest) goal of 
keeping tabular data in a form that anyone can read (and is likely to be able to read in 
the future) without needing to use a particular application: the data is in text form, 
one row per line, with entries separated by commas. You can find on the booksite 


domain 

key 

value 

phone 

book 

name 

phone 

number 

dictionary 

word 

definition 

account 

account 

number 

balance 

genomics 

codon 

amino acid 

data 

data/time 

results 

compiler 

variable 

name 

memory 

location 

file share 

song name 

machine 


website 

IP address 


Typical dictionary applications 
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% more ami no.csv 
TTT,Phe,F,Phenylalanine 
TTC,Phe,F,Phenylal ani ne 
TTA,Leu,L,Leuci ne 
TTC,Leu,L,Leuci ne 
TCT,Ser,S,Serine 
TCC,Ser,S,Serine 

CAA,Cly,C,Glutamic Acid 
CAC,Cly,C,Glutamic Acid 
CCT,Cly,C,Glycine 
CCC,Cly,C,Glycine 
CCA,Cly,C,Glycine 
CGC,Cly,C,Glycine 

% more DlIA.csv 

20-0ct-87,1738.74,608099968,1841.01 
19-Oct-87,2164.16,604300032,1738.74 
16-Oct-87,2355.09,338500000,2246.73 
15-Oct-87,2412.70,263200000,2355.09 

30-Oct-29,230.98,10730000,258.47 
29-Oct-29,252.38,16410000,230.07 
28-Oct-29,295.18,9210000,260.64 
25-Oct-29,299.47,5920000,301.22 


% more ip.csv 

www.ebay.com,66.135.192.87 
www.princeton.edu,128.112.128.15 
www.es.princeton.edu,128.112.136.35 
www.harvard.edu,128.103.60.24 
www.yale.edu,130.132.51.8 
www.enn.com,64.236.16.20 
www.google.com,216.239.41.99 
www.nytimes.com,199.239.136.200 
www.apple.com,17.112.152.32 
www.slashdot.org,66.35.250.151 
www.espn.com,199.181.135.201 
www.weather.com,63.111.66.11 
www.yahoo.com,216.109.118.65 


% more UPC.csv 

0002058102040,,”1 1/4"" STANDARD STORM DOOR" 
0002058102057,,"1 1/4"" STANDARD STORM DOOR" 
0002058102125,."DELUXE STORM DOOR UNIT" 
0002082012728,"100/ per box","12 gauge shells" 
0002083110812,"Classical CD","'Bits and Pieces'" 
002083142882,CD,"Garth Brooks - Ropin' The Wind" 
0002094000003,LB,"PATE PARISIEN" 

0002098000009,LB,"PATE TRUFFLE COCNAC-M&H 8Z RW" 
0002100001086,"16 oz","Kraft Parmesan" 
0002100002090,"15 pieces”,"Wrigley's Cum" 
0002100002434,"One pint","Trader Joe's milk" 


Typical comma-separated-value (.csv) files 


numerous . csv files that are related to vari¬ 
ous applications that we have described, 
including ami no. csv (codon-to-amino- 
acid encodings), DlIA.csv (opening price, 
volume, and closing price of the Dow Jones 
Industrial Average, for every day in its his¬ 
tory), ip.csv (a selection of entries from 
the DNS database), and upc.csv (the Uni¬ 
form Product Code bar codes that are wide¬ 
ly used to identify consumer products). 
Spreadsheet and other data-processing 
applications programs can read and write 
.csv files, and our example illustrates that 
you can also write a Java program to process 
the data any way that you would like. 

LookupCSV (on the facing page) builds a 
set of key-value pairs from a file of comma- 
separated values as specified on the com¬ 
mand line and then prints out values corre¬ 
sponding to keys read from standard input. 
The command-line arguments are the file 
name and two integers, one specifying the 
field to serve as the key and the other speci¬ 
fying the field to serve as the value. 

The purpose of this example is to il¬ 
lustrate the utility and flexibility of the 
symbol-table abstraction. What website 
has IP address 128.112.136.35? (www. 
cs .princeton. edu) What amino acid cor¬ 
responds to the codon TCA? (Seri ne) What 
was the DJIA on October 29,1929? (2 52.38) 
What product has UPC 0002100001086? 
(Kraft Parmesan) You can easily look up 
the answers to questions like these with 
LookupCSV and the appropriate . csv files. 

Performance is not much of an issue 
when handling interactive queries (since 
your computer can look through millions 
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Dictionary lookup 


public class LookupCSV 

{ 

public static void main(String[] args) 

{ 

In in = new ln(args[0]); 

int keyField = Integer.parselnt(args[1]); 

int valField = Integer.parselnt(args[2]); 

ST<String, String> st = new ST<String, String>(); 

while (in.hasNextLine()) 

{ 

String line = in.readLine(); 

String[] tokens = 1ine.split(","); 

String key = tokens[keyField]; 

String val = tokens[valField]; 
st.putfkey, val); 

} 

while (! Stdln. i sEmptyO) 

{ 

String query = Stdln.readString(); 
if (st.contains(query)) 

StdOut.println(st.getCquery)); 

} 

} 

} 


This data-driven symbol-table client reads key-value pairs from a file, then prints the values cor¬ 
responding to the keys found on standard input. Both keys and values are strings. The separating 
delimiter is taken as a command-line argument. 


% java LookupCSV ip.csv 1 0 

128.112.136.35 

www.cs.princeton.edu 


% java LookupCSV DDIA.csv 0 3 

29-0ct-29 

230.07 


% java LookupCSV amino.csv 0 3 

TCC 

Seri ne 


% java LookupCSV UPC.csv 0 2 
0002100001086 
Kraft Parmesan 
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of things in the time it takes to type a query), so fast implementations of ST are not no¬ 
ticeable when you use LookupCSV. However, when a program is doing the lookups (and 
a huge number of them), performance matters. For example, an internet router might 
need to look up millions of IP addresses per second. In this book, we have already seen 
the need for good performance with FrequencyCounter, and we will see several other 
examples in this section. 

Examples of similar but more sophisticated test clients for .csv files are described 
in the exercises. For instance, we could make the dictionary dynamic by also allowing 
standard-input commands to change the value associated with a key, or we could allow 
range searching, or we could build multiple dictionaries for the same file. 


Indexing clients Dictionaries are char¬ 
acterized by the idea that there is one value 
associated with each key, so the direct use of 
our ST data type, which is based on the asso¬ 
ciative-array abstraction that assigns one value 
to each key, is appropriate. Each account num¬ 
ber uniquely identifies a customer, each UPC 
uniquely identifies a product, and so forth. In 
general, of course, there may be multiple val¬ 
ues associated with a given key. For example, in 
our ami no. csv example, each codon identifies 
one amino acid, but each amino acid is asso¬ 
ciated with a list of codons, as in the example 
ami nol. csv at right, where each line contains 
an amino acid and the list of codons associated 
with it. We use the term index to describe sym¬ 
bol tables that associate multiple values with 
each key. Here are some more examples: 

■ Commercial transactions. One way for 


L 


aminoI.txt 

AT anine,AAT,AAC,CCT,CCC , CCA,GCC 
Arginine,CGT,CCC,CCA,CCC,ACA,AGC 
Aspartic Acid,CAT,CAC 
Cysteine,TCT,TCC 
Glutamic Acid,GAA,GAG 
Clutamine,CAA,CAC 
Clycine,CCT,GCC,CCA,CGC 
Histidine,CAT,CAC 
Isoleucine,ATT,ATC,ATA 
Leuci ne, TTA, TTC, CTT, CTC, CTA ,'CTG 
Lysine,AAA,AAC 
Methionine,ATC 
Phenylalanine,TTT,TTC 
Proline,CCT,CCC,CCA,CCC 
Serine,TCT,TCA,TCG,ACT,AGC 
Stop,TAA,TAG,TCA 
Th reonine,ACT,ACC,ACA,ACC 
Tyrosine,TAT,TAC 
T ryptophan,TGC 
Valine,CTT,CTC,CTA,CTG 

f '"V-"' 

key values 

A small index file (20 lines) 


separator 


a company that maintains customer 

accounts to keep track of a day’s transactions is to keep an index of the day’s 
transactions. The key is the account number; the value is the list of occurrences 
of that account number in the transaction list. 


■ Web search. When you type a keyword and get a list of websites containing that 
keyword, you are using an index created by your web search engine. There is one 
value (the set of pages) associated with each key (the query), although the reality 
is a bit more complicated because we often specify multiple keys. 
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■ Movies and performers. The file movi es . txt on the booksite (excerpted below) 
is taken from the Internet Movie Database (IMDB). Each line has a movie name 
(the key), followed by a list of performers in that movie (the value), separated by 
slashes. 

We can easily build an index by putting the values to be associated with each key into a 
single data structure (a Queue, say) and then associating that key with that data struc¬ 
ture as value. Extending LookupCSV along these lines is straightforward, but we leave 


that as an exercise (see exercise 3.5.12) 
and consider instead Lookuplndex on 

domain 

key 

value 

page 499, which uses a symbol table to build 
an index from files like ami nol . txt and 

genomics 

amino acid 

list of codons 

movies.txt (where the separator char- 

commercial 

account number 

list of transactions 

acter need not be a comma, as in a .csv 
file, but can be specified on the com- 

web search 

search key 

list of web pages 

mand line). After building the index, 

IMDB 

movie 

list of performers 

Lookuplndex then takes key queries and 
prints the values associated with each key. 
More interesting, Lookuplndex also builds 


Typical indexing applications 


an inverted index associated with each file, where values and keys switch roles. In the 
amino acid example, this gives the same functionality as Lookup (find the amino acid 
associated with a given codon); in the movie-performer example it adds the ability to 
find the movies associated with any given performer, which is implicit in the data but 
would be difficult to produce without a symbol table. Study this example carefully, as it 
provides good insight into the essential nature of symbol tables. 


movies.txt 

"/" separator 

Tin Men (1987)/DeBoy, David/Biumenfeid, Alan/... / 

Tirez sur le pianiste (1960)/Heymann, Claude/... Y 
Titanic (1997)/Mazin, Stan/...DiCaprio, Leonardo/... 

Titus (1999)/Weisskopf, Hermann/Rhys, Matthew/... 

To Be or Not to Be (1942)/Verebes, Erno (I)/... 

To Be or Not to Be (1983)/.../Brooks, Mel (I)/... 

To Catch a Thief (1955)/Paris, Manuel/... 

To Die For (1995)/Smith, Kurtwood/.../Kidman, Nicole/... 



key values 

Small portion of a large index file (250,000+ lines) 
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Inverted index. The term inverted index is normally applied to a situation where values 
are used to locate keys. We have a large amount of data and want to know where certain 
keys of interest occur. This application is another prototypical example of a symbol- 
table client that uses an intermixed sequence of calls to get O and put (). Again, we as¬ 
sociate each key with a SET of locations, where the occurrences of the key can be found. 
The nature and use of the location depend on the application: in a book, a location 
might be a page number; in a program, a location might be a line number; in genomics, 
a location might be a position in a genetic sequence; and so forth: 

■ Internet Movie DataBase (IMDB). In the example just considered, the input is 
an index that associates each movie with a list of performers. The inverted index 
associates each performer with a list of movies. 

■ Book index. Every textbook has an index where you look up a term and get 
the page numbers containing that term. While creating a good index generally 
involves work by the book author to eliminate common and irrelevant words, 


a document preparation 
system will certainly use a 

domain 

key 

value 

IMDB 

performer 

set of movies 

symbol table to help auto- 

mate the process. An interest¬ 
ing special case is known as a 

book 

term 

set of pages 

concordance, which associates 

compiler 

identifier 

set of places used 

each word in a text with the 
set of positions in the text 

file search 

search term 

set of files 

where that word occurs (see 

genomics 

subsequence 

set of locations 

exercise 3 . 5 . 20 ). 


Typical inverted indices 


■ Compiler. In a large program 

that uses a large number of symbols, it is useful to know where each name is 
used. Historically, an explicit printed symbol table was one of the most impor¬ 
tant tools used by programmers to keep track of where symbols are used in their 
programs. In modern systems, symbol tables are the basis of software tools that 
programmers use to manage names. 

■ File search. Modern operating systems provide you with the ability to type a term 
and to learn the names of files containing that term. The key is the term; the 
value is the set of files containing that term. 

■ Genomics. In a typical (if oversimplified) scenario in genomics research, a 
scientist wants to know the positions of a given genetic sequence in an existing 
genome or set of genomes. Existence or proximity of certain sequences may be 
of scientific significance. The starting point for such research is an index like a 
concordance, but modified to take into account the fact that genomes are not 
separated into words (see exercise 3 . 5 . 15 ). 
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Index (and inverted index) lookup 


public class Lookuplndex 

{ 

public static void main(String[] args) 

{ 

In in = new ln(args[0]); // index database 
String sp = args[l]; // separator 

ST<String, Queue<Stri ng» st = new ST<String, Queue<Stri ng»(); 
ST<String, Queue<Stri ng» ts = new ST<String, Queue<Stri ng»(); 


while (in.hasNextLineQ) 


{ 

String[] a = in.readLine().split(sp); 

String key = a[0]; 

for (int i =1; i < a.length; i++) 

{ 

String val = a[i]; 

if (!st.contains(key)) st.put(key, new Queue<String>()); 
if (!ts.contains(val)) ts.put(val, new Queue<String>()); 
st.get(key).enqueue(val); 
ts.get(val).enqueue(key); 

} 

} 

while (!Stdln.isEmptyO) 

{ 

String query = Stdln.readLine(); 
if (st.contains(query)) 

for (String s : st.get(query)) 

StdOut.println(" " + s); 

if (ts.contains(query)) 

for (String s : ts.get(query)) 

StdOut.println(" " + s); 


% java Lookuplndex aminoI.txt 
Seri ne 
TCT 
TCA 
TCG 
AGT 
AGC 
TCG 
Seri ne 


} 


} 


} 


This data-driven symbol-table client reads key-value pairs 
from a file, then prints the values corresponding to the keys 
found on standard input. Keys are strings; values are lists 
of strings. The separating delimiter is taken as a command¬ 
line argument. 


% java Lookuplndex movies.txt "/" 
Bacon, Kevin 

Mystic River (2003) 

Friday the 13th (1980) 
Flatliners (1990) 

Few Good Men, A (1992) 

Tin Men (1987) 

Blumenfeld, Alan 
DeBoy, David 
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Filelndex (on the facing page) takes file names from the command line and uses a 
symbol table to build an inverted index associating every word in any of the files with 
a SET of file names where the word can be found, then takes keyword queries from 
standard input, and produces its associated list of files. This process is similar to that 
used by familiar software tools for searching the web or for searching for information 
on your computer; you type a keyword to get a list of places where that keyword occurs. 
Developers of such tools typically embellish the process by paying careful attention to 

■ The form of the query 

■ The set of files/pages that are indexed 

■ The order in which files are listed in the response 

For example, you are certainly used to typing queries that contain multiple keywords 
to a web search engine (which is based on indexing a large fraction of the pages on the 
web) that provides answers in order of relevance or importance (to you or to an adver¬ 
tiser). The exercises at the end of this section address some of these embellishments. We 
will consider various algorithmic issues related to web search later, but the symbol table 
is certainly at the heart of the process. 

As with Lookuplndex, you are certainly encouraged to download Filelndex from 
the booksite and use it to index some text files on your computer or some websites of 
interest, to gain further appreciation for the utility of symbol tables. If you do so, you 
will find that it can build large indices for huge files with little delay, because each put 
operation and get request is taken care of immediately. Providing this immediate re¬ 
sponse for huge dynamic tables is one of the classic triumphs of algorithmic technology. 
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File indexing 


import java.io.Fi 1 e; 

public class Filelndex 

{ 

public static void main(String[] args) 

{ 

ST<String, SET<File» st = new ST<String, SET<File»(); 

for (String filename : args) 

{ 

File file = new Fi1e(fi1ename); 

In in = new In(file); 
while (!in.isEmptyO) 

{ 

String word = in.readString(); 

if (!st.contains(word)) st.put(word, new SET<Fi1e>()); 
SET<File> set = st.get(word); 
set.add(fi1e); 

} 

} 


} 


while (!Stdln.isEmptyO) 

{ 

String guery = Stdln. readStringO ; 
if (st.contains(query)) 

for (File file : st.get(query)) 

StdOut.println(" " + fi 1 e.getNameO) ; 

} 

} 


This symbol-table client indexes a set of files. We search for each word in each file in a symbol table, 
maintaining a SET of file names that contain the word. Names for In can also refer to web pages, so 
this code can also be used to build an inverted index of web pages. 


% more exl.txt 

% java Filelndex ex*.txt 

it was the best of times 

age 


ex3.txt 

% more ex2.txt 

ex4.txt 

it was the worst of times 

best 


exl.txt 

% more exB.txt 


it was the age of wisdom 

was 

exl.txt 

% more ex4.txt 

ex2.txt 

it was the age of foolishness 

ex3.txt 


ex4.txt 
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Sparse vectors Our next example illustrates the importance of symbol tables in sci¬ 
entific and mathematical calculations. We describe a fundamental and familiar calcula¬ 
tion that becomes a bottleneck in typical practical applications, then show how using a 
symbol table can remove the bottleneck and enable solution of vastly larger problems. 
Indeed, this particular calculation was at the core of the PageRank algorithm that was 
developed by S. Brin and L. Page and led to the emergence of Google in the early 2000s 

(and is a well-known mathematical abstraction 
that is useful in many other contexts). 

The basic calculation that we consider is ma¬ 
trix-vector multiplication: given a matrix and a 
vector, compute a result vector whose zth entry 
is the dot product of the given vector and the zth 
row of the matrix. For simplicity, we consider the 
case when the matrix is square with N rows and 
N columns and the vectors are of size N. This 
operation is elementary to code in Java, requir¬ 
ing time proportional to N 2 , for the N multipli¬ 
cations to compute each of the N entries in the 
result vector, which also matches the space proportional to N 2 that is required to store 
the matrix. 

In practice, it is very often the case that N is huge. For example, in the Google appli¬ 
cation cited above, N is the number of pages on the web. At the time PageRank was de¬ 
veloped, that was in the tens or hundreds of billions and it has skyrocketed since, so the 
value of N 2 would be far more than 10 20 . No one can afford that much time or space, so 
a better algorithm is needed. 

Fortunately, it is also often the 
case that the matrix is sparse: a huge 
number of its entries are 0. Indeed, 
for the Google application, the av¬ 
erage number of nonzero entries 
per row is a small constant: virtual¬ 
ly all web pages have links to only a 
few others (not all the pages on the 
web). Accordingly, we can represent 
the matrix as an array of sparse vec¬ 
tors, using a SparseVector imple¬ 
mentation like the HashST client on 

the facing page. Instead of using the Standard implementation of matrix _ vector multiplication 


doublet] [] a = new double[N] [N] ; 
doublet] x = new double[N]; 
doublet] b = new double[N]; 

// Initialize a[][] and x[] . 

for (int i =0; i < N; i++) 

{ 

sum = 0.0; 

for (int j = 0; j < N; j++) 
sum += a[i ] [j]*xtj] ; 
b[i] = sum; 


a[][] x[] b[] 


0 .90 0 0 o" 


.05~ 


.036 ” 

0 0 .36 .36 .18 


.04 


.297 

0 0 0 .90 0 


.36 

= 

.333 

.90 0 0 0 0 


.37 


.045 

.47 0 .47 0 0 


.19 


.1927 


Matrix-vector multiplication 
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Sparse vector with dot product 


public class SparseVector 

{ 

private HashST<Integer, Double> st; 

public SparseVectorO 
{ st = new HashST<Integer, Double>0; } 

public int sizeO 
{ return st.size(); } 

public void put(int i, double x) 

{ st.putfi, x) ; } 

public double get(int i) 

{ 

if (!st.contains(i)) return 0.0; 
else return st.get(i); 

} 

public double dot(double[] that) 

{ 

double sum = 0.0; 
for (int i : st.keysO) 

sum += that[i]*this.get(i); 
return sum; 

} 

} 


This symbol-table client is a bare-bones sparse vector implementation that illustrates an efficient 
dot product for sparse vectors. We multiply each entry by its counterpart in the other operand and 
add the result to a running sum. The number of multiplications required is equal to the number of 
nonzero entries in the sparse vector. 
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array of doubl e [] objects array of SparseVector objects 



Sparse matrix representations 


code a [i ] [ j ] to refer to the element in row i and column j, we use a [i ] . put (j , val) 
to set a value in the matrix and a [i ]. get (j ) to retrieve a value. As you can see from the 
code below, matrix-vector multiplication using this class is even simpler than with the 
array representation (and it more clearly describes the computation). More important, 
it only requires time proportional to N plus the number of nonzero elements in the 
matrix. 

For small matrices or matrices that are not sparse, the overhead for maintaining 
symbol tables can be substantial, but it is worth your while to be sure to understand 
the ramifications of using symbol tables for huge sparse matrices. To fix ideas, consider 
a huge application (like the one faced by Brin and 
Page) where Nis 10 billion or 100 billion, but the 
average number of nonzero elements per row is 
less than 10. For such an application, using symbol 
tables speeds up matrix-vector multiplication by a 
factor of a billion or more. The elementary nature 
of this application should not detract from its im¬ 
portance: programmers who do not take advan¬ 
tage of the potential to save time and space in this 
way severely limit their potential to solve practical 
problems, while programmers who do take factor- 

Sparse matrix-vector multiplication 


SparseVector[] a; 
a = new SparseVector[N]; 
doublet] x = new double[N] ; 
doublet] b = new double[N] ; 

// Initialize a[] and x[]. 

for (int i =0; i < N; i++) 
b [i ] = a [i ] . dot(x) ; 
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of-a-billion speedups when they are available are likely to be able to address problems 
that could not otherwise be contemplated. 

Building the matrix for the Google application is a graph-processing application 
(and a symbol-table client!), albeit for a huge sparse matrix. Given the matrix, the Page- 
Rank calculation is nothing more than doing a matrix-vector multiplication, replacing 
the source vector with the result vector, and iterating the process until it converges (as 
guaranteed by fundamental theorems in probability theory). Thus, the use of a class 
like SparseVector can improve the time and space usage for this application by a fac¬ 
tor of 10 billion or 100 billion or more. 

Similar savings are possible in many scientific calculations, so sparse vectors and ma¬ 
trices are widely used and typically incorporated into specialized systems for scientific 
computing. When working with huge vectors and matrices, it is wise to run simple per¬ 
formance tests to be sure that the kinds of performance gains that we have illustrated 
here are not being missed. On the other hand, array processing for primitive types of 
data is built in to most programming languages, so using arrays for vectors that are 
not sparse, as we did in this example, may offer further speedups. Developing a good 
understanding of the underlying costs and making the appropriate implementation 
decisions is certainly worthwhile for such applications. 


SYMBOL TABLES ARE A PRIMARY CONTRIBUTION OF ALGORITHMIC TECHNOLOGY to the 
development of otir modern computational infrastructure because of their ability to 
deliver savings on a huge scale in a vast array of practical applications, making the dif¬ 
ference between providing solutions to a wide range of problems and not being able 
to address them at all. Few fields of science or engineering involve studying the effects 
of an invention that improves costs by factors of 100 billion—symbol-table applica¬ 
tions put us in just that position, as we have just seen in several examples, and these 
improvements have had profound effects. The data structures and algorithms that we 
have considered are certainly not the final word: they were all developed in just a few 
decades, and their properties are not fully understood. Because of their importance, 
symbol-table implementations continue to be studied intensely by researchers around 
the world, and we can look forward to new developments on many fronts as the scale 
and scope of the applications they address continue to expand. 
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j Q&A 


Q. Can a SET contain null? 

A. No. As with symbol tables, keys are non-null objects. 

Q. Can a SET be null? 

A. No. A SET can be empty (contain no objects), but not nul 1. As with any Java data 
type, a variable of type SET can have the value nul 1, but that just indicates that it does 
not reference any SET. The result of using new to create a SET is always an object that is 
not nul1. 

Q. If all my data is in memory, there is no real reason to use a filter, right? 

A. Right. Filtering really shines in the case when you have no idea how much data to 
expect. Otherwise, it may be a useful way of thinking, but not a cure-all. 

Q. I have data in a spreadsheet. Can I develop something like LookupCSV to search 
through it? 

A. Your spreadsheet application probably has an option to export to a . csv file, so you 
can use LookupCSV directly. 

Q. Why would I need Fi 1 elndex? Doesn’t my operating system solve this problem? 

A. If you are using an OS that meets your needs, continue to do so, by all means. As 
with many of our programs, Fi 1 elndex is intended to show you the basic underlying 
mechanisms of such applications and to suggest possibilities to you. 

Q. Why not have the dot() method in SparseVector take a SparseVector object as 
argument and return a SparseVector object? 

A. That is a fine alternate design and a nice programming exercise that requires code 
that is a bit more intricate than for our design (see exercise 3 . 5 . 16 ). For general matrix 
processing, it might be worthwhile to also add a SparseMatri x type. 
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EXERCISES 


3 . 5.1 Implement SET and HashSET as “wrapper class” clients of ST and HashST, respec¬ 
tively (provide dummy values and ignore them). 

3 . 5.2 Develop a SET implementation Sequenti al SearchSET by starting with the code 
for Sequenti al SearchST and eliminating all of the code involving values. 

3 . 5.3 Develop a SET implementation Bi narySearchSET by starting with the code for 
Bi narySearchST and eliminating all of the code involving values. 

3 . 5.4 Develop classes HashSTint and HashSTdouble for maintaining sets of keys of 
primitive int and double types, respectively. (Convert generics to primitive types in 
the code of Li nearProbi ngHashST.) 

3 . 5.5 Develop classes STint and STdouble for maintaining ordered symbol ta¬ 
bles where keys are primitive int and double types, respectively. (Convert generics 
to primitive types in the code of RedBl ackBST.) Test your solution with a version of 
SparseVector as a client. 

3 . 5.6 Develop classes HashSETi nt and HashSETdoubl e for maintaining sets of keys of 
primitive i nt and doubl e types, respectively. (Eliminate code involving values in your 
solution to exercise 3 . 5 . 4 .) 

3 . 5.7 Develop classes SETint and SETdouble for maintaining ordered sets of keys of 
primitive i nt and doubl e types, respectively. (Eliminate code involving values in your 
solution to exercise 3 . 5 . 5 .) 

3 . 5.8 Modify Li nearProbi ngHashST to keep duplicate keys in the table. Return any 
value associated with the given key for get 0 , and remove all items in the table that have 
keys equal to the given key for deleteO. 

3 . 5.9 Modify BST to keep duplicate keys in the tree. Return any value associated with 
the given key for get(), and remove all nodes in the tree that have keys equal to the 
given key for del ete (). 

3 . 5.10 Modify RedBl ackBST to keep duplicate keys in the tree. Return any value associ¬ 
ated with the given key for get (), and remove all nodes in the tree that have keys equal 
to the given key for del ete Q. 
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EXERCISES (continued) 


3 . 5.11 Develop a Mul ti SET class that is like SET, but allows equal keys and thus imple¬ 
ments a mathematical multiset. 

3 . 5.12 Modify LookupCSV to associate with each key all values that appear in a key- 
value pair with that key in the input (not just the most recent, as in the associative-array 
abstraction). 

3 . 5.13 Modify LookupCSV to make a program RangeLookupCSV that takes two key val¬ 
ues from the standard input and prints all key-value pairs in the . csv file such that the 
key falls within the range specified. 

3 . 5.14 Develop and test a static method invertO that takes as argument an 
ST<Stri ng, Bag<Stri ng» and produces as return value the inverse of the given sym¬ 
bol table (a symbol table of the same type). 

3 . 5.15 Write a program that takes a string on standard input and an integer k as com¬ 
mand-line argument and puts on standard output a sorted list of the k-grams found in 
the string, each followed by its index in the string. 

3 . 5.16 Add a method sum() to SparseVector that takes a SparseVector as argument 
and returns a SparseVector that is the term-by-term sum of this vector and the argu¬ 
ment vector. Note: You need del ete () (and special attention to precision) to handle the 
case where an entry becomes 0. 
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CREATIVE PROBLEMS 


3 . 5.17 Mathematical sets. Your goal is to develop an implementation of the following 
API MathSET for processing (mutable) mathematical sets: 

public class MathSET<Key> 



MathSETfKey[] universe) 

create a set 

voi d 

add(Key key) 

put key into the set 

MathSET<Key> 

compl ementO 

set of keys in the universe that 
are not in this set 

voi d 

uni on(MathSET<Key> a) 

put any keys from a into the 
set that are not already there 

voi d 

intersection(MathSET<Key> a) 

remove any keys from this set 
that are not in a 

voi d 

delete(Key key) 

remove key from the set 

boolean 

contains(Key key) 

is key in the set? 

boolean 

isEmptyO 

is the set empty? 

i nt 

size() 

number of keys in the set 


API for a basic set data type 



Use a symbol table. Extra credit : Represent sets with arrays of bool ean values. 

3 . 5.18 Multisets. After referring to exercises 3 . 5.2 and 3 . 5.3 and the previous exer¬ 
cise, develop APIs Mul ti HashSET and Mul ti SET for multisets (sets that can have equal 
keys) and implementations SeparateChai ni ngMul ti SET and Bi narySearchMul ti SET 
for multisets and ordered multisets, respectively. 

3 . 5.19 Equal keys in symbol tables. Consider the API Mul ti ST (unordered or ordered) 
to be the same as our symbol-table APIs defined on page 363 and page 366, but with equal 
keys allowed, so that the semantics of get() is to return any value associated with the 
given key, and we add a new method 

Iterable<Value> getAll(Key key) 
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CREATIVE PROBLEMS (continued) 


that returns all values associated with the given key. Using our code for 
SeparateChai ni ngST and Bi narySearchST as a starting point, develop implementa¬ 
tions SeparateChai ni ngMul ti ST and Bi narySearchMul ti ST for these APIs. 

3.5.20 Concordance. Write an ST client Concordance that puts on standard output a 
concordance of the strings in the standard input stream (see page 498). 

3.5.21 Inverted concordance. Write a program InvertedConcordance that takes a 
concordance on standard input and puts the original string on standard output stream. 
Note: This computation is associated with a famous story having to do with the Dead 
Sea Scrolls. The team that discovered the original tablets enforced a secrecy rule that 
essentially resulted in their making public only a concordance. After a while, other re¬ 
searchers figured out how to invert the concordance, and the full text was eventually 
made public. 

3.5.22 Fully indexed CSV. Implement an ST client Ful 1 LookupCSV that builds an ar¬ 
ray of ST objects (one for each field), with a test client that allows the user to specify the 
key and value fields in each query. 

3.5.23 Sparse matrices. Develop an API and an implementation for sparse 2D matri¬ 
ces. Support matrix addition and matrix multiplication. Include constructors for row 
and column vectors. 

3.5.24 Non-overlapping interval search. Given a list of non-overlapping intervals of 
items, write a function that takes an item as argument and determines in which, if 
any, interval that item lies. For example, if the items are integers and the intervals are 
1643-2033, 5532-7643, 8999-10332, 5666653-5669321, then the query point 9122 
lies in the third interval and 8122 lies in no interval. 

3.5.25 Registrar scheduling. The registrar at a prominent northeastern University re¬ 
cently scheduled an instructor to teach two different classes at the same exact time. Help 
the registrar prevent future mistakes by describing a method to check for such conflicts. 
For simplicity, assume all classes run for 50 minutes starting at 9:00,10:00,11:00, 1:00, 
2:00, or 3:00. 

3.5.26 LRU cache. Create a data structure that supports the following operations: ac¬ 
cess and remove. The access operation inserts the item onto the data structure if it’s 
not already present. The remove operation deletes and returns the item that was least 
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recently accessed. Hint: Maintain the items in order of access in a doubly linked list, 
along with pointers to the first and last nodes. Use a symbol table with keys = items, 
values = location in linked list. When you access an element, delete it from the linked 
list and reinsert it at the beginning. When you remove an element, delete it from the end 
and remove it from the symbol table. 

3.5.27 List. Develop an implementation of the following API: 
public class List<Item> implements Iterable<Item> 



ListO 

create a list 

voi d 

addFront(Item item) 

add i tem to the front 

voi d 

addBack(Item item) 

add i tem to the back 

Item 

del eteFrontO 

remove from the front 

Item 

deleteBack() 

remove from the back 

voi d 

delete(Item item) 

remove i tem from the list 

voi d 

add(int i, Item item) 

add i tem as the i th in the list 

Item 

delete(int i) 

remove the i th item from the list 

boolean 

contains(Item item) 

is key in the list? 

boolean 

isEmpty() 

is the list empty? 

i nt 

sizeQ 

number of items in the list 


API for a list data type 

Hint: Use two symbol tables, one to find the i th item in the list efficiently, and the other 
to efficiently search by item. (Java’s java, uti 1 . List interface contains methods like 
these but does not supply any implementation that efficiently supports all 
operations.) 

3.5.28 UniQueue. Create a data type that is a queue, except that an element may only 
be inserted the queue once. Use an existence symbol table to keep track of all elements 
that have ever been inserted and ignore requests to re-insert such items. 
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CREATIVE PROBLEMS (continued) 


3.5.29 Symbol table with random access. Create a data type that supports inserting a 
key-value pair, searching for a key and returning the associated value, and deleting and 
returning a random key. Hint: Combine a symbol table and a randomized queue. 
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EXPERIMENTS 


3.5.30 Duplicates (revisited). Redo exercise 2 . 5.31 using the Dedup filter given on 
page 490. Compare the running times of the two approaches. Then use Dedup to run the 
experiments for N= 10 7 , 10 8 , andlO 9 , repeat the experiments for random long values 
and discuss the results. 

3.5.31 Spell checker. With the file di cti onary. txt from the booksite as command¬ 
line argument, the B1 ac k Filter client described on page 491 prints all misspelled words 
in a text file taken from standard input. Compare the performance of RedBlackBST, 
SeparateChai ni ngHashST, and Li nearProbi ngHashST for the file WarAndPeace. txt 
(available on the booksite) with this client and discuss the results. 

3.5.32 Dictionary. Study the performance of a client like LookupCSV in a scenario 
where performance matters. Specifically, design a query-generation scenario instead of 
taking commands from standard input, and run performance tests for large inputs and 
large numbers of queries. 

3.5.33 Indexing. Study a client like Lookuplndex in a scenario where performance 
matters. Specifically, design a query-generation scenario instead of taking commands 
from standard input, and run performance tests for large inputs and large numbers of 
queries. 

3.5.34 Sparse vector. Run experiments to compare the performance of matrix-vector 
multiplication using SparseVector to the standard implementation using arrays. 

3.5.35 Primitive types. Evaluate the utility of using primitive types for Integer and 
Double values, for Li nearProbi ngHashST and RedBlackBST. How much space and 
time are saved, for large numbers of searches in large tables? 
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P airwise connections between items play a critical role in a vast array of compu¬ 
tational applications. The relationships implied by these connections lead im¬ 
mediately to a host of natural questions: Is there a way to connect one item to 
another by following the connections? How many other items are connected to a given 
item? What is the shortest chain of connections between this item and this other item? 

To model such situations, we use abstract mathematical objects called graphs. In this 
chapter, we examine basic properties of graphs in detail, setting the stage for us to study 
a variety of algorithms that are useful for answering questions of the type just posed. 
These algorithms serve as the basis for attacking problems in important applications 
whose solution we could not even contemplate without good algorithmic technology. 

Graph theory, a major branch of mathematics, has been studied intensively for hun¬ 
dreds of years. Many important and useful properties of graphs have been discovered, 
many important algorithms have been developed, and many difficult problems are still 
actively being studied. In this chapter, we introduce a variety of fundamental graph 
algorithms that are important in diverse applications. 

Like so many of the other problem domains that we have studied, the algorithmic in¬ 
vestigation of graphs is relatively recent. Although a few of the fundamental algorithms 
are centuries old, the majority of the interesting ones have been discovered within the 
last several decades and have benefited from the emergence of the algorithmic technol¬ 
ogy that we have been studying. Even the simplest graph algorithms lead to useful com¬ 
puter programs, and the nontrivial algorithms that we examine are among the most 
elegant and interesting algorithms known. 

To illustrate the diversity of applications that involve graph processing, we begin our 
exploration of algorithms in this fertile area by introducing several examples. 
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Maps. A person who is planning a trip may need to answer questions such as “What is 
the shortest route from Providence to Princeton?” A seasoned traveler who has experi¬ 
enced traffic delays on the shortest route may ask the question “What is the fastest way 
to get from Providence to Princeton?” To answer such questions, we process informa¬ 
tion about connections (roads) between items (intersections). 

Web content. When we browse the web, we encounter pages that contain references 
(links) to other pages and we move from page to page by clicking on the links. The 
entire web is a graph, where the items are pages and the connections are links. Graph¬ 
processing algorithms are essential components of the search engines that help us lo¬ 
cate information on the web. 

Circuits. An electric circuit comprises devices such as transistors, resistors, and ca¬ 
pacitors that are intricately wired together. We use computers to control machines that 
make circuits and to check that the circuits perform desired functions. We need to an¬ 
swer simple questions such as “Is a short-circuit present?” as well as complicated ques¬ 
tions such as “Can we lay out this circuit on a chip without making any wires cross?” 
The answer to the first question depends on only the properties of the connections 
(wires), whereas the answer to the second question requires detailed information about 
the wires, the devices that those wires connect, and the physical constraints of the chip. 

Schedules. A manufacturing process requires a variety of jobs to be performed, under 
a set of constraints that specify that certain tasks cannot be started until certain other 
tasks have been completed. How do we schedule the tasks such that we both respect the 
given constraints and complete the whole process in the least amount of time? 

Commerce. Retailers and financial instututions track buy/sell orders in a market. A 
connection in this situation represents the transfer of cash and goods between an in¬ 
stitution and a customer. Knowledge of the nature of the connection structure in this 
instance may enhance our understanding of the nature of the market. 

Matching. Students apply for positions in selective institutions such as social clubs, 
universities, or medical schools. Items correspond to the students and the institutions; 
connections correspond to the applications. We want to discover methods for matching 
interested students with available positions. 

Computer networks. A computer network consists of interconnected sites that send, 
forward, and receive messages of various types. We are interested in knowing about the 
nature of the interconnection structure because we want to lay wires and build switches 
that can handle the traffic efficiently. 
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Software. A compiler builds graphs to represent relationships among modules in a 
large software system. The items are the various classes or modules that comprise the 
system; connections are associated either with the possibility that a method in one class 
might call another (static analysis) or with actual calls while the system is in operation 
(dynamic analysis). We need to analyze the graph to determine how best to allocate 
resources to the program most efficiently. 

Social networks. When you use a social network, you build explicit connections with 
your friends. Items correspond to people; connections are to friends or followers. Un¬ 
derstanding the properties of these networks is a modern graph-processing applica¬ 
tions of intense interest not just to compaines that support such networks, but also in 
politics, diplomacy, entertainment, education, marketing, and many other domains. 

these examples indicate the range of applications for which graphs are the ap¬ 
propriate abstraction and also the range of computational problems that we might 
encounter when we work with graphs. Thousands of such problems have been studied, 
but many problems can be addressed in the context of one of several basic graph mod¬ 
els—we will study the most important 


ones in this chapter. In practical appli¬ 
cations, it is common for the volume of 

application 

item 

connection 

map 

intersection 

road 

data involved to be truly huge, so that 

efficient algorithms make the difference 
between whether or not a solution is at 

web content 

page 

link 

all feasible. 

circuit 

device 

wire 

To organize the presentation, we 

schedule 

job 

constraint 

progress through the four most impor¬ 
tant types of graph models: undirected 
graphs (with simple connections), di¬ 
graphs (where the direction of each con¬ 
nection is significant), edge-weighted 

commerce 

matching 

computer network 

transaction 

customer 

student 

site 

application 

connection 

graphs (where each connection has an 

software 

method 

call 

associated weight), and edge-weighted 
digraphs (where each connection has 

social network 

person 

friendship 

both a direction and a weight). 

Typical graph applications 
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4.1 UNDIRECTED GRAPHS 



our starting point is the study of graph models where edges are nothing more than 
connections between vertices. We use the term undirected graph in contexts where we 
need to distinguish this model from other models (such as the title of this section), but, 
since this is the simplest model, we start with the following definition: 

Definition. A graph is a set of vertices and a collection of edges that each connect a 
pair of vertices. 


Vertex names are not important to the definition, but we need a way 
to refer to vertices. By convention, we use the names 0 through V— 1 
for the vertices in a V-vert ex graph. The main reason that we choose 
this system is to make it easy to write code that efficiently accesses in¬ 
formation corresponding to each vertex, using array indexing. It is not 
difficult to use a symbol table to establish a 1-1 mapping to associate 
V arbitrary vertex names with the V integers between 0 and V— 1 (see 
page 548), so the convenience of using indices as vertex names comes 
without loss of generality (and without much loss of efficiency). We 
use the notation v-w to refer to an edge that connects v and w; the nota¬ 
tion w-v is an alternate way to refer to the same edge. 

We draw a graph with circles for the vertices and lines connecting 
them for the edges. A drawing gives us intuition about the structure of 
the graph; but this intuition can be misleading, because the graph is 
defined independently of the drawing. For example, the two drawings 
at left represent the same graph, because the graph is nothing more than its (unor¬ 
dered) set of vertices and its (unordered) collection of edges (vertex pairs). 

Anomalies. Our definition allows two simple anomalies: 

■ A self-loop is an edge that connects a vertex to itself. 

■ Two edges that connect the same pair of vertices are parallel. 

Mathematicians sometimes refer to graphs with parallel edges 
as multigraphs and graphs with no parallel edges or self-loops as 
simple graphs. Typically, our implementations allow self-loops and 
parallel edges (because they arise in applications), but we do not include them in ex¬ 
amples. Thus, we can refer to every edge just by naming the two vertices it connects. 



Anomalies 



Two drawings of the same graph 
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Glossary A substantial amount of nomenclature is associated with graphs. Most of 
the terms have straightforward definitions, and, for reference, we consider them in one 
place: here. 

When there is an edge connecting two vertices, we say that the vertices are adjacent 
to one another and that the edge is incident to both vertices. The degree of a vertex is the 
number of edges incident to it. A subgraph is a subset of a graph’s edges (and associated 
vertices) that constitutes a graph. Many computational tasks 
involve identifying subgraphs of various types. Of particular 
interest are edges that take us through a sequence of vertices 
in a graph. 


Most often, we work with simple cycles and simple paths and 
drop the simple modifer; when we want to allow repeated ver¬ 
tices, we refer to general paths and cycles. We say that one vertex is connected to another 
if there exists a path that contains both of them. We use notation like u-v-w-x to repre¬ 
sent a path from u to x and u-v-w-x-u to represent a cycle from u to v to w to x and back 
to u again. Several of the algorithms that we consider find paths and cycles. Moreover, 
paths and cycles lead us to consider the structural properties of a graph as a whole: 

Definition. A graph is connected if there is a path from every vertex to every other 
vertex in the graph. A graph that is not connected consists of a set of connected com¬ 
ponents, which are maximal connected subgraphs. 


Definition. A path in a graph is a sequence of vertices 
connected by edges. A simple path is one with no repeated 
vertices. A cycle is a path with at least one edge whose first 
and last vertices are the same. A simple cycle is a cycle with 
no repeated edges or vertices (except the requisite repeti¬ 
tion of the first and last vertices). The length of a path or 
a cycle is its number of edges. 


vertex 



Intuitively, if the vertices were physical objects, such as knots or beads, and the edges 
were physical connections, such as strings or wires, a connected graph would stay in 
one piece if picked up by any vertex, and a graph that is not connected comprises two or 
more such pieces. Generally, processing a graph necessitates processing the connected 
components one at a time. 
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19 vertices 



sparse (E = 200) 


An acyclic graph is a graph with no cycles. Several of 
the algorithms that we consider are concerned with find¬ 
ing acyclic subgraphs of a given graph that satisfy certain 
properties. We need additional terminology to refer to 
these structures: 

Definition. A tree is an acyclic connected graph. A dis¬ 
joint set of trees is called a forest. A spanning tree of a 
connected graph is a subgraph that contains all of that 
graph’s vertices and is a single tree. A spanning forest of 
a graph is the union of spanning trees of its connected 
components. 


This definition of tree is quite general: with suitable refine¬ 
ments it embraces the trees that we typically use to model pro¬ 
gram behavior (function-call hierarchies) and data structures 
(BSTs, 2-3 trees, and so forth). Mathematical properties of 
trees are well-studied and intuitive, so we state them without 
proof. For example, a graph G with V vertices is a tree if and 
only if it satisfies any of the following five conditions: 

■ G has V— 1 edges and no cycles. 

■ G has V— 1 edges and is connected. 

■ G is connected, but removing any edge disconnects it. 

■ G is acyclic, but adding any edge creates a cycle. 

■ Exactly one simple path connects each pair of vertices in G. 

Several of the algorithms that we consider find spanning trees and forests, and these 
properties play an important role in their analysis and implementation. 

The density of a graph is the propor- 



dense (£=1000) 




Two graphs (V= 50) 


tion of possible pairs of vertices that are 
connected by edges. A sparse graph has 
relatively few of the possible edges pres¬ 
ent; a dense graph has relatively few of 
the possible edges missing. Generally, 
we think of a graph as being sparse if 
its number of different edges is within 
a small constant factor of V and as be¬ 
ing dense otherwise. This rule of thumb 
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leaves a gray area (when the number of edges is, say, ~ c V 312 ) but the distinction be¬ 
tween sparse and dense is typically very clear in applications. The applications that we 
consider nearly always involve sparse graphs. 

A bipartite graph is a graph whose vertices we can divide into two sets 
such that all edges connect a vertex in one set with a vertex in the other 
set. The figure at right gives an example of a bipartite graph, where one 
set of vertices is colored red and the other set of vertices is colored black. 

Bipartite graphs arise in a natural way in many situations, one of which 
we will consider in detail at the end of this section. 



with these preparations, we are ready to move on to consider graph-processing 
algorithms. We begin by considering an API and implementation for a graph data type, 
then we consider classic algorithms for searching graphs and for identifying connected 
components. To conclude the section, we consider real-world applications where vertex 
names need not be integers and graphs may have huge numbers of vertices and edges. 
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Undirected graph data type Our starting point for developing graph-process¬ 
ing algorithms is an API that defines the fundamental graph operations. This scheme 
allows us to address graph-processing tasks ranging from elementary maintenance op¬ 
erations to sophisticated solutions of difficult problems. 

public class Graph 

Graph (i nt V) create a V -vertexgraph with no edges 

Graph (In i n) read a graph from input stream i n 

V () number of vertices 

E() number of edges 

addEdge(int v, int w) add edge v-w to this graph 
adjfint v) vertices adjacent to v 

toSt r i ng () string representation 

API for an undirected graph 

This API contains two constructors, methods to return the number of vertices and 
edges, a method to add an edge, a toStri ng() method, and a method adj () that al¬ 
lows client code to iterate through the vertices adjacent to a given vertex (the order of 
iteration is not specified). Remarkably, we can build all of the algorithms that we con¬ 
sider in this section on the basic abstraction embodied in adj (). 

The second constructor assumes an input format consisting of 2E + 2 integer values: 
V, then E, then E pairs of values between 0 and V—1, each pair denoting an edge. As 
examples, we use the two graphs ti nyG. txt and medi umG. txt that are depicted below. 

Several examples of Graph client code are shown in the table on the facing page. 


int 
int 
voi d 

Iterable<Integer> 
String 


V- 


tinyG.txt 



V. 


mediumG.txt 


250 

1273 ^. 

244 246 
239 240 
238 245 
235 238 
233 240 
232 248 
231 248 
229 249 
228 241 
226 231 


E 



(1263 additional lines) 


Input format for Graph constructor (two examples) 
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task 

implementation 

compute the degree ofv 

public static int degree(Graph G, int v) 

{ 

int degree = 0; 

for (int w : G.adj(v)) degree++; 
return degree; 

} 

compute maximum degree 

public static int maxDegree(Graph G) 

{ 

int max = 0; 

for (int v = 0; v < G.V(); v++) 
if (degree(G, v) > max) 
max = degree(G, v); 
return max; 

} 

compute average degree 

public static int avgDegree(Graph G) 

{ return 2 * G.E() / G.V(); } 

count self-loops 

public static int numberOfSelfLoops(Graph G) 

{ 

int count = 0; 

for (int v = 0; v < G.V(); v++) 
for (int w : G.adj(v)) 
if (v == w) count++; 

return count/2; // each edge counted twice 

} 

string representation of the 
graph’s adjacency lists 
(instance method in Graph) 

public String toStringO 
{ 

String s = V + " vertices, " + E + " edges\n 
for (int v = 0; v < V; v++) 

{ 

s += v + ": "; 
for (int w : this.adj(v)) 
s += w + " "; 
s += "\n"; 

} 

return s; 

} 

Typical graph-processing code 
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Representation alternatives. The next decision that we face in graph processing is 
which graph representation (data structure) to use to implement this API. We have two 
basic requirements: 

■ We must have the space to accommodate the types of graphs that we are likely to 
encounter in applications. 

■ We want to develop time-efficient implementations of Graph instance meth¬ 
ods—the basic methods that we need to develop graph-processing clients. 

These requirements are a bit vague, but they 
are still helpful in choosing among the three 
data structures that immediately suggest 
themselves for representing graphs: 

■ An adjacency matrix, where we main¬ 
tain a V-by-V boolean array, with the 
entry in row v and column w defined 
to be true if there is an edge adjacent 
to both vertex v and vertex w in the 
graph, and to be f al se otherwise. This 
representation fails on the first count— 
graphs with millions of vertices are 
common and the space cost for the V 2 
boolean values needed is prohibitive. 

■ An array of edges, using an Edge class 
with two instance variables of type i nt. 

This direct representation is simple, 


but it fails on the second count— 
implementing ad j () would involve 
examining all the edges in the graph. 
An array of adjacency lists, where we 
maintain a vertex-indexed array of lists 
of the vertices adjacent to each vertex. 
This data structure satisfies both re¬ 
quirements for typical applications and 
is the one that we will use throughout 
this chapter. 



Adjacency-lists representation (undirected graph) 


Beyond these performance objectives, a detailed examination reveals other consider¬ 
ations that can be important in some applications. For example, allowing parallel edges 
precludes the use of an adjacency matrix, since the adjacency matrix has no way to 
represent them. 
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Adjacency-lists data structure. The standard graph representation for graphs that are 
not dense is called the adjacency-lists data structure, where we keep track of all the 
vertices adjacent to each vertex on a linked list that is associated with that vertex. We 
maintain an array of lists so that, given a vertex, we can immediately access its list. To 
implement lists, we use our Bag ADT from section 1.3 with a linked-list implementa¬ 
tion, so that we can add new edges in constant time and iterate through adjacent verti¬ 
ces in constant time per adjacent vertex. The Graph implementation on page 526 is based 
on this approach, and the figure on the facing page depicts the data structures built by 
this code for ti nyG. txt. To add an edge connecting v and w, we add w to v’s adjacency 
list and v to w’s adjacency list. Thus, each edge appears twice in the data structure. This 
Graph implementation achieves the following performance characteristics: 

■ Space usage proportional to V + E 

■ Constant time to add an edge 

■ Time proportional to the degree of v to iterate through vertices adjacent to v 
(constant time per adjacent vertex processed) 

These characteristics are optimal for this set of operations, which suffice for the graph¬ 
processing applications that we consider. Parallel edges and self-loops are allowed (we 
do not check for them). Note: It is important to realize that the order in which edges 
are added to the graph determines the order in which vertices appear in the array of 
adjacency lists built by Graph. Many different ar¬ 
rays of adjacency lists can represent the same graph. 

When using the constructor that reads edges from 
an input stream, this means that the input format 
and the order in which edges are specified in the 
file determine the order in which vertices appear 
in the array of adjacency lists built by Graph. Since 
our algorithms use ad j () and process all adjacent 
vertices without regard to the order in which they 
appear in the lists, this difference does not affect 
their correctness, but it is important to bear it in 
mind when debugging or following traces. To fa¬ 
cilitate these activities, we assume that Graph has a 
test client that reads a graph from the input stream 
named as command-line argument and then prints 
it (relying on the toStringO implementation on 
page 523) to showthe order in which vertices appear 
in adjacency lists, which is the order in which algo¬ 
rithms process them (see exercise 4 . 1 . 7 ). 



tinyG.txt 

0 5 

4 3 
01 
912 

6 4 

5 4 
0 2 
11 12 
9 10 
0 6 

7 8 
9 11 
5 3 


% java Graph tinyG.txt 
13 vertices, 13 edges 


0 

1 

2 

3 

4 

5 

6 

7 

8 

9 

10 
11 
12 


S' 


first adjacent 
vertex in input 


last on list 


6 2 1 5 ( 

0 
0 

5 4 
5 
3 
0 4 
8 

7 

11 10 12 

n second 

representation 
9 12 of each edge 

11 9 appears in red 


Output for list-of-edges input 
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Graph data type 


public class Graph 

{ 

private final int V; 

private int E; 

private Bag<Integer>[] adj 


// number of vertices 
// number of edges 
// adjacency lists 


public Graph(int V) 

{ 

this.V = V; this.E = 0; 
adj = (Bag<Integer>[]) new Bag[V] 
for (int v = 0; v < V; v++) 
adj [v] = new Bag<lnteger>0 ; 

} 

public Graph(In in) 

{ 

this(i n. readlntO) ; 
int E = in. readlntO ; 
for (int i =0; i < E; i++) 

{ // Add an edge. 

int v = in. readlntO; 
int w = in.readlntO; 
addEdge(v, w) ; 


// Create array of lists. 
// Initialize all lists 
// to empty. 


// Read V and construct this graph. 
// Read E. 


// Read a vertex, 

// read another vertex, 

// and add edge connecting them. 


} 


} 


public int V() 
public int E() 


return V; 
return E; 


} 


public void addEdge(int v, int w) 

{ 

adj[v].add(w); 
adj[w].add(v); 

E++; 

} 

public Iterable<Integer> adj(int v) 
{ return adj[v]; } 


// Add w to v’s list. 
// Add v to w’s list. 


This Graph implementation maintains a vertex-indexed array of lists of integers. Every edge appears 
twice: if an edge connects v and w, then w appears in v’s list and v appears in w’s list. The second con¬ 
structor reads a graph from an input stream, in the format V followed by E followed by a list of pairs 
of i nt values between 0 and V—\. See page 523 for toStri ng() . 
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it is certainly reasonable to contemplate other operations that might be useful in 
applications, and to consider methods for 

■ Adding a vertex 

■ Deleting a vertex 

One way to handle such operations is to expand the API to use a symbol table (ST) 
instead of a vertex-indexed array (with this change we also do not need our convention 
that vertex names be integer indices). We might also consider methods for 

■ Deleting an edge 

■ Checking whether the graph contains the edge v-w 

To implement these methods (and disallow parallel edges) we might use a SET instead 
of a Bag for adjacency lists. We refer to this alternative as an adjacency set representa¬ 
tion. We do not use these alternatives in this book for several reasons: 

■ Our clients do not need to add vertices, delete vertices and edges, or check 
whether an edge exists. 

■ When clients do need these operations, they typically are invoked infrequently 
or for short adjacency lists, so an easy option is to use a brute-force implementa¬ 
tion that iterates through an adjacency list. 

■ The SET and ST representations slightly complicate algorithm implementation 
code, diverting attention from the algorithms themselves. 

■ A performance penalty of log V may be involved in some situations. 

It is not difficult to adapt our algorithms to accommodate other designs (for example 
disallowing parallel edges or self-loops) without undue performance penalties. The 
table below summarizes performance characteristics of the alternatives that we have 
mentioned. Typical applications process huge sparse graphs, so we use the adjacency- 
lists representation throughout. 


underlying 
data structure 

space 

add edge v-w 

check whether w is 
adjacent to v 

iterate through vertices 
adjacent to v 

list of edges 

E 

1 

E 

E 

adjacency matrix 

V 2 

1 

1 

y 

adjacency lists 

E+V 

1 

degreety ) 

degreety') 

adjacency sets 

E+V 

i°gy 

logT 

log y + degree{\/) 


Order-of-growth performance for typical Graph implementations 
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Design pattern for graph processing. Since we consider a large number of graph-pro¬ 
cessing algorithms, our initial design goal is to decouple our implementations from the 
graph representation. To do so, we develop, for each given task, a task-specific class so 
that clients can create objects to perform the task. Generally, the constructor does some 
preprocessing to build data structures so as to efficiently respond to client queries. A 
typical client program builds a graph, passes that graph to an algorithm implementa¬ 
tion class (as argument to a constructor), and then calls client query methods to learn 
various properties of the graph. As a warmup, consider this API: 


public class Search 


boolean 
int 


Search (Graph G, int s) find vertices connected to a source vertex s 

marked(int v) is v connected to s? 

count () how many vertices are connected to s? 


Graph-processing API (warmup) 


We use the term source to distinguish the vertex provided as argument to the construc¬ 
tor from the other vertices in the graph. In this API, the job of the constructor is to find 
the vertices in the graph that are connected to the source. Then client code calls the in¬ 
stance methods marked () and count() to learn characteristics of the graph. The name 
marked () refers to an approach used by the basic algorithms that we consider through¬ 
out this chapter: they follow paths from the source to other vertices in the graph, mark¬ 
ing each vertex encountered. The example client TestSearch shown on the facing page 
takes an input stream name and a source vertex number from the command line, reads 
a graph from the input stream (using the second Graph constructor), builds a Search 
object for the given graph and source, and uses marked O to print the vertices in that 
graph that are connected to the source. It also calls count () and prints whether or not 
the graph is connected (the graph is connected if and only if the search marked all of 
its vertices). 
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we have already seen one way to implement the Search API: the union-find algo¬ 
rithms of chapter i. The constructor can build a UF object, do a uni on 0 operation 
for each of the graph’s edges, and implement marked (v) by calling connected (s , v). 
Implementing count () requires using a weighted UF implementation and extending 
its API to use a count () method that returns wt[fi nd(v)] (see exercise 4 . 1 . 8 ). This 
implementation is simple and efficient, but the implementation that we consider next 
is even simpler and more efficient. It is based on depth-first search, a fundamental recur¬ 
sive method that follows the graph’s edges to find the vertices connected to the source. 
Depth-first search is the basis for several of the graph-processing algorithms that we 
consider throughout this chapter. 


public class TestSearch 

{ 

public static void main(String[] args) 

{ 

Graph G = new Graph(new ln(args[0])); 
int s = Integer.parselnt(args[1]); 
Search search = new Search(G, s); 

for (int v = 0; v < G.V(); v++) 
if (search.marked(v)) 

StdOut. print(v + " ") ; 

StdOut. pri ntl n() ; 

if (search.count() != G.V()) 

StdOut.print("NOT "); 

StdOut.println("connected"); 

} 

} 

Sample graph-processing client (warmup) 


% java TestSearch tinyG.txt 0 
0 1 2 3 4 5 6 
NOT connected 

% java TestSearch tinyG.txt 9 
9 10 11 12 
NOT connected 
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Depth-first search We often learn properties of a graph by systematically examin¬ 
ing each of its vertices and each of its edges. Determining some simple graph proper¬ 
ties—for example, computing the degrees of all the vertices—is easy if we just exam¬ 
ine each edge (in any order whatever). But many other graph properties are related to 
paths, so a natural way to learn them is to move from vertex to vertex along the graph’s 
edges. Nearly all of the graph-processing algorithms 
that we consider use this same basic abstract model, 
albeit with various different strategies. The simplest 
is a classic method that we now consider. 






Equivalent models of a maze 



Searching in a maze. It is instructive to think 
about the process of searching through a graph in 
terms of an equivalent problem that has a long and 
distinguished history—finding our way through a 
maze that consists of passages connected by inter¬ 
sections. Some mazes can be handled with a simple 
rule, but most mazes require a more sophisticated 
strategy. Using the terminology maze instead of 
graph, passage instead of edge, and intersection in¬ 
stead of vertex is making mere semantic distinc¬ 
tions, but, for the moment, doing so will help to 
give us an intuitive feel for the problem. One trick for exploring 
a maze without getting lost that has been known since antiquity 
(dating back at least to the legend of Theseus and the Minotaur) is 
known as Tremaux exploration. To explore all passages in a maze: 

■ Take any unmarked passage, unrolling a string behind you. 

■ Mark all intersections and passages when you first visit 
them. 

■ Retrace steps (using the string) when approaching a marked 
intersection. 

■ Retrace steps when no unvisited options remain at an inter¬ 
section encountered while retracing steps. 

The string guarantees that you can always find a way out and the 
marks guarantee that you avoid visiting any passage or intersection twice. Knowing 
that you have explored the whole maze demands a more complicated argument that is 
better approached in the context of graph search. Tremaux exploration is an intuitive 
starting point, but it differs in subtle ways from exploring a graph, so we now move on 
to searching in graphs. 




Tremaux exploration 
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Warmup. The classic recursive method for 
searching in a connected graph (visiting all 
of its vertices and edges) mimics Tremaux 
maze exploration but is even simpler to de¬ 
scribe. To search a graph, invoke a recursive 
method that visits vertices. To visit a vertex: 

■ Mark it as having been visited. 

■ Visit (recursively) all the vertices that 
are adjacent to it and that have not 
yet been marked. 

This method is called depth-first search 
(DFS). An implementation of our Search 
API using this method is shown at right. 
It maintains an array of boolean val¬ 
ues to mark all of the vertices that are 
connected to the source. The recursive 
method marks the given vertex and calls 
itself for any unmarked vertices on its 
adjacency list. If the graph is connect¬ 
ed, every adjacency-list entry is checked. 


public class DepthFi rstSearch 
{ 

private boolean[] marked; 
private int count; 

public DepthFirstSearch(Craph C, int s) 
{ 

marked = new boolean[C.V()]; 
dfsCC, s); 

} 

private void dfs(Graph G, int v) 

{ 

marked[v] = true; 
count++; 

for (int w : G.adj(v)) 

if (!marked[w]) dfs(G, w) ; 

} 

public boolean marked(int w) 

{ return marked[w]; } 

public int count() 

{ return count; } 


Depth-first search 


Proposition A. DFS marks all the vertices connected to a 
given source in time proportional to the sum of their degrees. 

Proof: First, we prove that the algorithm marks all the verti¬ 
ces connected to the source s (and no others). Every marked 
vertex is connected to s, since the algorithm finds vertices 
only by following edges. Now, suppose that some unmarked 
vertex w is connected to s. Since s itself is marked, any path 
from s to w must have at least one edge from the set of marked 
vertices to the set of unmarked vertices, say v-x. But the al¬ 
gorithm would have discovered x after marking v, so no such 
edge can exist, a contradiction. The time bound follows be¬ 
cause marking ensures that each vertex is visited once (taking 
time proportional to its degree to check marks). 


source set of marked 
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tinyCG.txt 

V 


standard drawing 



drawing with both edges 



adjacency lists 


One-way passages. The method call-return mechanism in the program corresponds 
to the string in the maze: when we have processed all the edges incident to a vertex 
(explored all the passages leaving an intersection), we “return” (in both senses of the 
word). To draw a proper correspondence with Tremaux exploration of a maze, we need 
to imagine a maze constructed entirely of one-way passages (one in each direction). 

In the same way that we encounter each passage 
in the maze twice (once in each direction), we 
encounter each edge in the graph twice (once at 
each of its vertices). In Tremaux exploration, we 
either explore a passage for the first time or re¬ 
turn along it from a marked vertex; in DFS of 
an undirected graph, we either do a recursive 
call when we encounter an edge v-w (if w is not 
marked) or skip the edge (if w is marked). The 
second time that we encounter the edge, in the 
opposite orientation w-v, we always ignore it, 
because the destination vertex v has certainly al¬ 
ready been visited (the first time that we encoun¬ 
tered the edge). 

Tracing DFS. As usual, one good way to under¬ 
stand an algorithm is to trace its behavior on a 
small example. This is particularly true of depth- 
first search. The first thing to bear in mind when 
doing a trace is that the order in which edges 
are examined and vertices visited depends upon 
the representation, not just the graph or the al¬ 
gorithm. Since DFS only examines vertices con¬ 
nected to the source, we use the small connected 
graph depicted at left as an example for traces. 

In this example, vertex 2 is the first vertex visited 
after 0 because it happens to be first on 0’s adjacency list. The second thing to bear in 
mind when doing a trace is that, as mentioned above, DFS traverses each edge in the 
graph twice, always finding a marked vertex the second time. One effect of this obser¬ 
vation is that tracing a DFS takes twice as long as you might think! Our example graph 
has only eight edges, but we need to trace the action of the algorithm on the 16 entries 
on the adjacency lists. 



A connected undirected graph 
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Detailed trace of depth-first search. The figure at right shows the contents of the data 


structures just after each vertex is marked for our small example, with source 0. The 


search begins when the constructor calls the 
recursive dfsO to mark and visit vertex 0 
and proceeds as follows: 

■ Since 2 is first on 0 ’s adjacency list 
and is unmarked, df s O recursively 
calls itself to mark and visit 2 (in ef¬ 
fect, the system puts 0 and the current 
position on 0 ’s adjacency list on a 
stack). 

■ Now, 0 is first on 2’s adjacency list 
and is marked, so dfs() skips it. 

Then, since 1 is next on 2’s adjacency 
list and is unmarked, dfs() recur¬ 
sively calls itself to mark and visit 1 . 

■ Visiting 1 is different: since both ver¬ 
tices on its list (0 and 2 ) are already 
marked, no recursive calls are needed, 
and df s O returns from the recursive 
call df s (1) . The next edge examined 
is 2-3 (since 3 is the vertex after 1 on 
2 ’s adjacency list), so dfsO recur¬ 
sively calls itself to mark and visit 3. 

■ Vertex 5 is first on 3’s adjacency list 
and is unmarked, so df s () recursively 
calls itself to mark and visit 5. 

■ Both vertices on 5’s list (3 and 0) are 
already marked, so no recursive calls 
are needed, 

■ Vertex 4 is next on 3’s adjacency list 
and is unmarked, so df s () recursively 
calls itself to mark and visit 4, the last 
vertex to be marked. 

■ After 4 is marked, dfs() needs to 
check the vertices on its list, then the 


dfs(0) 


dfs(2) 
check 0 


dfs(l) 
check 0 
check 2 
1 done 


dfs(3) 


dfs(5) 
check 3 
check 0 
5 done 


dfs(4) 
check 3 
check 2 
4 done 
check 2 
3 done 
check 4 
2 done 
check 1 
check 5 
0 done 



marked[] adj[] 

° I T 0 12 15 
1 10 2 

2 2 0 1 3 4 

3 3 5 4 2 

4 4 3 2 

5 I 5 I 3 0 


0 I T 0 12 15 
1 10 2 

2 T 2 0 13 4 

3 3 5 4 2 

4 4 3 2 

5 I 5 I 3 0 


0 | T 0 12 15 

1 T 10 2 

2 ! T 2 13 4 

3 3 5 4 2 

4 4 3 2 

5 I 5 1 3 0 


0 I T 0 12 15 

1 T 10 2 

2 T 2 3 4 

3 T 3 5 4 2 

4 4 3 2 

5 I 5 I 3 0 


0 I T 0 12 15 

1 T 10 2 

2 T 2 0 13 4 

3 T 3 5 4 2 

4 4 3 2 

5 | T 5 3 0 


0 I T 0 12 15 

1 T 10 2 

2 T 2 0 13 4 

3 T 3 5 4 2 

4 T 4 3 2 

5 T 5 h 


Trace of depth-first search to find vertices connected to 0 


remaining vertices on 3’s list, then 2’s list, then 0’s list, but no more recursive 
calls happen because all vertices are marked. 
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this basic recursive scheme is just a start —depth-first search is effective for many 
graph-processing tasks. For example, in this section, we consider the use of depth-first 
search to address a problem that we first posed in chapter i: 

Connectivity. Given a graph, support queries of the form Are two given vertices 
connected ? and How many connected components does the graph have ? 

This problem is easily solved within our standard graph-processing design pattern, and 
we will compare and contrast this solution with the union-find algorithms that we 
considered in section 1 . 5 . 

The question “Are two given vertices connected?” is equivalent to the question “Is 
there a path connecting two given vertices?” and might be named the path detection 
problem. However, the union-find data structures that we considered in section 1.5 do 
not address the problems of finding such a path. Depth-first search is the first of several 
approaches that we consider to solve this problem, as well: 

Single-source paths. Given a graph and a source vertex s, support queries of the 
form Is there a path from s to a given target vertex v? If so, find such a path. 

DFS is deceptively simple because it is based on a familiar concept and is so easy to 
implement; in fact, it is a subtle and powerful algorithm that researchers have learned 
to put to use to solve numerous difficult problems. These two are the first of several that 
we will consider. 
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Finding paths The single-source paths problem is fundamental to graph process¬ 
ing. In accordance with our standard design pattern, we use the following API: 

public class Paths 

Paths (Graph G, int s) find paths in G from source s 
boolean hasPathTo(i nt v) is there a path from s to v? 

Iterabl e<Integer> pathTo(int v) path from s to v; null if no such path 

API for paths implementations 


The constructor takes a source vertex s as 
argument and computes paths from s to 
each vertex connected to s. After creating 
a Paths object for a source s, the client can 
use the instance method pathToO to iter¬ 
ate through the vertices on a path from s to 
any vertex connected to s. For the moment, 
we accept any path; later, we shall develop 
implementations that find paths having 
certain properties. The test client at right 
takes a graph from the input stream and a 
source from the command line and prints 
a path from the source to each vertex con¬ 
nected to it. 


public static void main(String[] args) 

{ 

Graph G = new Graph(new ln(args[0])); 
int s = Integer.parselnt(args[1]); 

Paths search = new Paths(G, s); 
for (int v = 0; v < G.V(); v++) 

{ 

StdOut. pri nt(s + " to " + v + ": "); 
if (search.hasPathTo(v)) 

for (int x : search.pathTo(v)) 
if (x == s) StdOut.print(x) ; 
else StdOut. pri nt("-" + x) ; 
StdOut .pri ntl n() ; 

} 

} 

Test client for paths implementations 


Implementation, algorithm 4.1 onpage 536 is a DFS-based implementation of Paths 
that extends the Depth Fi rstSearch warmup onpage 531 by adding as an instance vari¬ 
able an array edgeTo[] of i nt values that serves the purpose of the ball of string in 
Tremaux exploration: it gives a way to find a path back to s for every vertex connected 
to s. Instead of just keeping track of the path from the current vertex back to the start, 

we remember a path from each vertex to the 
start. To accomplish this, we remember the edge 
v-w that takes us to each vertex w for the first 
time, by setting edgeTo [w] to v. In other words, 
v-w is the last edge on the known path from s 
to w. The result of the search is a tree rooted at 
the source; edgeTo [] is a parent-link represen¬ 
tation of that tree. A small example is drawn to 


% java Paths tinyCG.txt 0 
0 to 0: 0 
0 to 1: 0-2-1 
0 to 2: 0-2 
0 to 3: 0-2-3 
0 to 4: 0-2-3-4 
0 to 5: 0-2-3-5 
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ALGORITHM 4.1 Depth-first search to find paths in a graph 


public class DepthFirstPaths 

{ 

private booleanf] marked; // Has dfs() been called for this vertex? 
private int[] edgeTo; // last vertex on known path to this vertex 

private final int s; // source 


public DepthFirstPaths(Graph G, int s) 

{ 

marked = new boolean[G.V()]; 
edgeTo = new int[G.V()]; 
this.s = s; 
dfs(G, s); 

} 


private void dfs(Graph G, int v) 

{ 

markedfv] = true; 
for (int w : G.adj(v)) 
if (!marked[w]) 

{ 

edgeTo[w] = v; 
dfs(G, w); 

} 

} 

public boolean hasPathTo(int v) 

{ return markedfv]; } 



edgeTo[] 

0 

1 2 
2 0 

3 2 

4 3 

5 3 



5 5 

3 3 5 

2 2 3 5 

0 0 2 3 5 

Trace of pathTo(5) computation 


public Iterable<Integer> pathTo(int v) 

{ 

if (!hasPathTo(v)) return null; 
Stack<Integer> path = new Stack<Integer>(); 
for (int x = v; x != s; x = edgeTo[x]) 
path.push(x); 
path.push(s) ; 
return path; 

} 


This Graph client uses depth-first search to find paths to all the vertices in a graph that are connected 
to a given start vertex s. Code from DepthFi rstSearch (page 531) is printed in gray. To save known 
paths to each vertex, this code maintains a vertex-indexed array edgeTo [] such that edgeTo [w] = v 
means that v-w was the edge used to access w for the first time. The edgeTo [] array is a parent-link 
representation of a tree rooted at s that contains all the vertices connected to s. 
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the right of the code in algorithm 4 . 1 . To recover 
the path from s to any vertex v, the pathToO method 
in algorithm 4.1 uses a variable x to travel up the 
tree, setting x to edgeTo [x] , just as we did for union- 
find in section 1 . 5 , putting each vertex encountered 
onto a stack until reaching s. Returning the stack to 
the client as an Iterabl e enables the client to follow 
the path from s to v. 

Detailed trace. The figure at right shows the con¬ 
tents of edgeTo [] just after each vertex is marked 
for our example, with source 0. The contents of 
marked [] and adj [] are the same as in the trace of 
DepthFi rstSearch on page 533, as is the detailed de¬ 
scription of the recursive calls and the edges checked, 
so these aspects of the trace are omitted. The depth- 
first search adds the edges 0-2, 2-1, 2-3, 3-5, and 
3-4 to edgeTo [], in that order. These edges form a 
tree rooted at the source and provide the information 
needed for pathToO to provide for the client the path 
from 0 to 1, 2, 3, 4, or 5, as just described. 

the constructor in DepthFi rstPaths differs only 
in a few assignment statements from the constructor 
in DepthFi rstSearch, so proposition a on page 531 
applies. In addition, we have: 

Proposition A (continued). DFS allows us to pro¬ 
vide clients with a path from a given source to any 
marked vertex in time proportional its length. 

Proof: By induction on the number of verti¬ 
ces visited, it follows that the edgeTo [] array in 
DepthFi rstPaths represents a tree rooted at the 
source. The pathToO method builds the path in 
time proportional to its length. 


dfs(0) 


dfs(2) 
check 0 


dfs(l) 
check 0 
check 2 
1 done 


dfs(3) 


dfs(5) 
check 3 
check 0 
5 done 


dfs(4) 
check 3 
check 2 
4 done 
check 2 
3 done 
check 4 
2 done 
check 1 
check 5 
0 done 








edgeTo[] 

0 

1 

2 

3 

4 

5 I 


0 I 

1 

2 0 

3 

4 
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0 I 
1 2 
2 0 

3 

4 
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1 2 
2 0 

3 2 

4 
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Trace of depth-first search to find all paths from 0 
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Breadth-first search The paths discovered by depth-first search depend not just 
on the graph, but also on the representation and the nature of the recursion. Naturally, 
we are often interested in solving the following problem: 

Single-source shortest paths. Given a graph and a source vertex s, support que¬ 
ries of the form Is there a path from s to a given target vertex v? If so, find a shortest 
such path (one with a minimal number of edges). 

The classical method for accomplishing this task, called breadth-first search (BFS), is 
also the basis of numerous algorithms for processing graphs, so we consider it in detail 
in this section. DFS offers us little assistance in solving this problem, because the order 
in which it takes us through the graph has no relationship to the goal of find¬ 
ing shortest paths. In contrast, BFS is based on this goal. To find a shortest path 
from s to v, we start at s and check for v among all the vertices that we can 
reach by following one edge, then we check for v among all the vertices that we 
can reach from s by following two edges, and so forth. DFS is analogous to one 
person exploring a maze. BFS is analogous to a group of searchers exploring by 
fanning out in all directions, each unrolling his or her own ball of string. When 
more than one passage needs to be explored, we imagine that the searchers 
split up to expore all of them; when two groups of searchers meet up, they join 
forces (using the ball of string held by the one getting there first). 

In a program, when we come to a point during a graph search where we 
have more than one edge to traverse, we choose one and save the others to be 
explored later. In DFS, we use a pushdown stack (that is managed by the sys¬ 
tem to support the recursive search method) for this purpose. Using the LIFO 
rule that characterizes the pushdown stack corresponds to exploring passages 
that are close by in a maze. We choose, of the passages yet to be explored, 
the one that was most recently encountered. In BFS, we want to explore the 
vertices in order of their distance from the source. It turns out that this order is easily 
arranged: use a (FIFO) queue instead of a (LIFO) stack. We choose, of the passages yet 
to be explored, the one that was least recently encountered. 

Implementation, algorithm 4.2 on page 540 is an implementation of BFS. It is based 
on maintaining a queue of all vertices that have been marked but whose adjacency lists 
have not been checked. We put the source vertex on the queue, then perform the fol¬ 
lowing steps until the queue is empty: 

■ Take the next vertex v from the queue and mark it. 

■ Put onto the queue all unmarked vertices that are adjacent to v. 
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The bfs() method in algorithm 4.2 is not re¬ 
cursive. Instead of the implicit stack provided by 
recursion, it uses an explicit queue. The product of 
the search, as for DFS, is an array edgeTo [], a par¬ 
ent-link representation of a tree rooted at s, which 
defines the shortest paths from s to every 



edgeTo[] 
0 



Outcome of breadth-first search to find all paths from 0 


vertex that is connected to s. The paths 
can be constructed for the client using the 
same pathToO implementation that we 
used for DFS in algorithm 4 . 1 . 

The figure at right shows the step-by- 
step development of BFS on our sample 
graph, showing the contents of the data 
structures at the beginning of each it¬ 
eration of the loop. Vertex 0 is put on the 
queue, then the loop completes the search 
as follows: 

■ Removes 0 from the queue and puts 
its adjacent vertices 2,1, and 5 on 
the queue, marking each and setting 
the edgeTo [] entry for each to 0. 

■ Removes 2 from the queue, checks 
its adjacent vertices 0 and 1 , which 
are marked, and puts its adjacent 
vertices 3 and 4 on the queue, mark¬ 
ing each and setting the edgeTo [] 
entry for each to 2 . 

■ Removes 1 from the queue and 
checks its adjacent vertices 0 and 2 , 
which are marked. 

■ Removes 5 from the queue and 
checks its adjacent vertices 3 and 0, 
which are marked. 

■ Removes 3 from the queue and 
checks its adjacent vertices 5, 4, 
and 2 , which are marked. 


queue 



marked [] edgeTo [] adj [] 
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Trace of breadth-first search to find all paths from 0 


■ Removes 4 from the queue and 

checks its adjacent vertices 3 and 2, which are marked. 
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ALGORITHM 4.2 Breadth-first search to find paths in a graph 


public class BreadthFirstPaths 

{ 

private booleanf] marked; // Is a shortest path to this vertex known? 
private int[] edgeTo; // last vertex on known path to this vertex 

private final int s; // source 

public BreadthFirstPaths(Graph G, int s) 

{ 

marked = new boolean[G.V()]; 
edgeTo = new int[G.V()]; 
this.s = s; 
bfs(G, s); 

} 


private void bfs(Graph G, int 

{ 

Queue<Integer> queue = new 
marked[s] = true; 
queue.enqueue(s) ; 
while (!q.isEmptyO) 

{ 

int v = queue.dequeue() 
for (int w : G.adj(v)) 
if (!marked[w]) 

{ 

edgeTo[w] = v; 
marked[w] = true; 
queue.enqueue(w); 

} 

} 

} 


s) 


Queue<Integer>(); 

// Mark the source 

// and put it on the queue. 


// Remove next vertex from the queue. 

// For every unmarked adjacent vertex, 

// save last edge on a shortest path, 
// mark it because path is known, 

// and add it to the queue. 


public boolean hasPathTo(int v) 
{ return marked[v]; } 


public Iterable<Integer> pathTo(int v) 

// Same as for DFS (see page 536). 


} 


This Graph client uses breadth-first search to find paths in a graph with the fewest number of edges 
from the source s given in the constructor. The bfs() method marks all vertices connected to s, so 
clients can use hasPathToO to determine whether a given vertex v is connected to s and pathToO to 
get a path from s to v with the property that no other such path from s to v has fewer edges. 
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For this example, the edgeTo[] array is complete after the second step. As with DFS, 
once all vertices have been marked, the rest of the computation is just checking edges to 
vertices that have already been marked. 

Proposition B. For any vertex v reachable from s, BFS computes a shortest path 
from s to v (no path from s to v has fewer edges). 

Proof: It is easy to prove by induction that the queue always consists of zero or 
more vertices of distance k from the source, followed by zero or more vertices of 
distance k+1 from the source, for some integer k, starting with k equal to 0. This 
property implies, in particular, that vertices enter and leave the queue in order of 
their distance from s. When a vertex v enters the queue, no shorter path to v will 
be found before it comes off the queue, and no path to v that is discovered after it 
comes off the queue can be shorter than v’s tree path length. 


Proposition B (continued). BFS takes time proportional to V+E in the worst case. 

Proof: As for proposition a (page 531), BFS marks all the vertices connected to s in 
time proportional to the sum of their degrees. If the graph is connected, this sum is 
the sum of the degrees of all the vertices, or 2 E. 


Note that we can also use BFS to implement the Search API that we implemented with 
DFS, since the solution depends on only the ability of the search to examine every ver¬ 
tex and edge connected to the source. 

As implied at the outset, DFS and BFS are the first of several instances that we will 
examine of a general approach to searching graphs. We put the source vertex on the 
data structure, then perform the following steps until the data structure is empty: 

■ Take the next vertex v from the data structure and mark it. 

■ Put onto the data structure all unmarked vertices that are adjacent to v. 

The algorithms differ only in the rule used to 
take the next vertex from the data structure 
(least recently added for BFS, most recently add¬ 
ed for DFS). This difference leads to completely 
different views of the graph, even though all the 
vertices and edges connected to the source are 
examined no matter what rule is used. 


% java 

Breadtl 

0 to 0 

0 

0 to 1 

0-1 

0 to 2 

0-2 

0 to 3 

0-2-3 

0 to 4 

0-2-4 

0 to 5 

0-5 
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20% 



DFS for paths (250 vertices) 


THE DIAGRAMS ON EITHER SIDE of 
this page, which show the progress of 
DFS and BFS for our sample graph 
mediumG.txt, make plain the differ¬ 
ences between the paths that are dis¬ 
covered by the two approaches.DFS 
wends its way through the graph, stor¬ 
ing on the stack the points where other 
paths branch off; BFS sweeps through 
the graph, using a queue to remember 
the frontier of visited places. DFS ex¬ 
plores the graph by looking for new 
vertices far away from the start point, 
taking closer vertices only when dead 
ends are encountered; BFS completely 
covers the area close to the starting 
point, moving farther away only when 
everything nearby has been examined. 
DFS paths tend to be long and wind¬ 
ing; BFS paths are short and direct. 
Depending upon the application, one 
property or the other may be desirable 
(or properties of paths may be imma¬ 
terial). In section 4 . 4 , we will be con¬ 
sidering other implementations of the 
Paths API that find paths having other 
specified properties. 



20 % 


BFS for shortest paths (250 vertices) 
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Connected components Our next direct application of depth-first search is to 
find the connected components of a graph. Recall from section i . 5 (see page 216) that “is 
connected to” is an equivalence relation that divides the vertices into equivalence classes 
(the connected components). For this common graph-processing task, we define the 
following API: 


public class CC 


CC(Craph C) 

boolean connectedfint v, int w) 
int count() 


int id(int v) 


preprocessing constructor 
are v and w connected? 
number of connected components 

component identifier for v 
(between 0 and count()-l) 

API for connected components 


The i d () method is for client use in indexing an array by component, as in the test cli¬ 
ent below, which reads a graph and then prints its number of connected components 
and then the vertices in each component, one component per line. To do so, it builds an 
array of Bag objects, then uses each vertex’s component identifier as an index into this 
array, to add the vertex to the appropriate Bag. This client is a model for the typical sit¬ 
uation where we want to independently 
process connected components. 

Implementation. The implementation 
CC (algorithm 4.3 on the next page) 
uses our marked [] array to find a vertex 
to serve as the starting point for a depth- 
first search in each component. The first 
call to the recursive DFS is for vertex 0— 
it marks all vertices connected to 0. Then 
the f o r loop in the constructor looks for 
an unmarked vertex and calls the recur¬ 
sive df s O to mark all vertices connected 
to that vertex. Moreover, it maintains a 
vertex-indexed array id[] that associ¬ 
ates the same int value to every vertex 
in each component. This array makes 
the implementation of connected() 

simple, in precisely the same manner as Test cMent for connected components API 


public static void main(String[] args) 

{ 

Graph G = new Graph(new In(args[0])); 

CC cc = new CC(G); 

int M = cc. count(); 

StdOut.println(M + " components"); 

Bag<Integer>[] components; 

components = (Bag<Integer>[]) new Bag[M]; 

for (int i =0; i < M; i++) 

components[i] = new Bag<Integer>(); 
for (int v = 0; v < G.V(); v++) 
components[cc.id(v)].add(v); 
for (int i =0; i < M; i++) 

{ 

for (int v: components[i]) 

StdOut.print(v + " "); 

StdOut. pri ntl n() ; 

} 
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ALGORITHM 4.3 Depth-first search to find connected components in a graph 


public class CC 

{ 

private booleanf] marked; 

private int[] id; 
private int count; 

public CC(Graph G) 

{ 

marked = new boolean[G.V()]; 

id = new int[G.V()]; 
for (int s = 0; s < G.V(); s++) 
if (!marked[s]) 

{ 

dfs(G, s); 
count++; 

} 

} 

private void dfs(Graph G, int v) 

{ 

markedfv] = true; 
id[v] = count; 
for (int w : G.adj(v)) 
if (!marked[w]) 
dfs(G, w); 

} 

public boolean connected(int v, int w) 
{ return id[v] == id[w]; } 

public int id(int v) 

{ return id[v] ; } 

public int countO 
{ return count; } 


% more tinyG.txt 
IB vertices, IB edges 
0: 6 2 1 5 
1 : 0 
2 : 0 
3: 5 4 
4: 5 6 3 
5:340 
6:0 4 
7: 8 
8: 7 

9: 11 10 12 
10: 9 
11: 9 12 
12: 11 9 

% java CC tinyG.txt 
3 components 
6 5 4 3 2 1 0 
8 7 

12 11 10 9 


} 


This Graph client provides its clients with the ability to independently process a graph’s connected 
components. Code from DepthFi rstSearch (page 531) is left in gray. The computation is based on 
a vertex-indexed array i d [] such that i d [v] is set to i if v is in the i th connected component pro¬ 
cessed. The constructor finds an unmarked vertex and calls the recursive dfs() to mark and identify 
all the vertices connected to it, continuing until all vertices have been marked and identified. Imple¬ 
mentations of the instance methods connected (), i d(), and countO are immediate. 
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tinyC.txt 



marked[] 


0123456789 101112 


dfs(0) 0 

dfs(6) 0 

check 0 

dfs(4) 0 

dfs(5) 0 

dfs(3) 0 

check 5 
check 4 
3 done 
check 4 
check 0 
5 done 
check 6 
check 3 


4 done 
6 done 

dfs(2) 0 

| check 0 
2 done 

dfs(l) 0 

check 0 
1 done 
check 5 
0 done 

dfs(7) 1 

dfs(8) 1 

check 7 
8 done 
7 done 

dfs(9) 2 

dfs(11) 2 

check 9 

dfs(12) 2 


check 11 
check 9 
12 done 
11 done 

dfs(10) 2 

check 9 
10 done 
check 12 
9 done 


T 

T T 

T T T 

T T T T 

T T T T T 


T T T T T T 


T T T T T T T 


TTTTTTTT 

TTTTTTTTT 


T T 
T T 

T T 


T T 
T T 

T T 


T T 
T T 

T T 


T T 
T T 

T T 


T T 
T T 

T T 


T 

T T 


TTTTTTTTTTTTT 


_ id[] 

01234567 

0 

0 0 

0 0 0 

0 0 0 0 

0 0 0 0 0 


0 0 0 0 0 0 


0 0 0 0 0 0 0 


00000001 

00000001 


00000001 

00000001 

00000001 


00000001 


Trace of depth-first search to find connected components 


8 9101112 


1 

1 2 

12 2 
12 2 2 

1 2 2 2 2 
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connected () in section 1.5 (just check if identifiers are equal). In this case, the identi¬ 
fier 0 is assigned to all the vertices in the first component processed, 1 is assigned to all 
the vertices in the second component processed, and so forth, so that the identifiers are 
all between 0 and count ()-l, as specified in the API. This convention enables the use 
of component-indexed arrays, as in the test client on page 543. 

Proposition C. DFS uses preprocessing time and space proportional to V+E to 
support constant-time connectivity queries in a graph. 

Proof: Immediate from the code. Each adjacency-list entry is examined exactly 
once, and there are 2 E such entries (two for each edge). Instance methods examine 
or return one or two instance variables. 


Union-find. How does the DFS-based solution for graph connectivity in CC compare 
with the union-find approach of chapter 1 ? In theory, DFS is faster than union-find 
because it provides a constant-time guarantee, which union-find does not; in practice, 
this difference is negligible, and union-find is faster because it does not have to build 
a full representation of the graph. More important, union-find is an online algorithm 
(we can check whether two vertices are connected in near-constant time at any point, 
even while adding edges), whereas the DFS solution must first preprocess the graph. 
Therefore, for example, we prefer union-find when determining connectivity is our 
only task or when we have a large number of queries intermixed with edge insertions 
but may find the DFS solution more appropriate for use in a graph ADT because it 
makes efficient use of existing infrastructure. 

the problems that we have solved with DFS are fundamental. It is a simple ap¬ 
proach, and recursion provides us a way to reason about the computation and develop 
compact solutions to graph-processing problems. Two additional examples, for solving 
the following problems, are given in the table on the facing page. 

Cycle detection. Support this query: Is a given graph acylicl 

Two-colorability. Support this query: Can the vertices of a given graph be assigned 
one of two colors in such a way that no edge connects vertices of the same color I 
which is equivalent to this question: Is the graph bipartite ? 

As usual with DFS, the simple code masks a more sophisticated computation, so study¬ 
ing these examples, tracing their behavior on small sample graphs, and extending them 
to provide a cycle or a coloring, respectively, are worthwhile (and left for exercises). 
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task 

implementation 

is C acyclic? 

(assumes 
no self-loops or 
parallel edges) 

public class Cycle 
{ 

private booleanf] marked; 

private boolean hasCycle; 

public Cycle(Craph C) 

{ 

marked = new boolean[G.V()]; 
for (int s = 0; s < G.V(); s++) 
if (!marked[s]) 
dfs(G, s, s); 

} 

private void dfs(Graph G, int v, int u) 

{ 

marked[v] = true; 
for (int w : G.adj(v)) 
if (!marked[w]) 
dfs(G, w, v); 

else if (w != u) hasCycle = true; 

} 

public boolean hasCycle() 

{ return hasCycle; } 

} 

is C bipartite? 

( two-colorable ) 

public class TwoColor 

{ 

private booleanf] marked; 

private booleanf] color; 

private boolean isTwoColorable = true; 

public TwoColor(Graph G) 

{ 

marked = new boolean[G.V()]; 
color = new boolean[G.V()]; 
for (int s = 0; s < G.V(); s++) 
if (!marked[s]) 
dfs(G, s); 

} 

private void dfs(Graph G, int v) 

{ 

marked[v] = true; 
for (int w : G.adj(v)) 
if (!marked[w]) 

{ 

colorfw] = !color[v]; 

dfs(G, w) ; 

} 

else if (col or [w] == col or [v]) isTwoColorable = false 

} 

public boolean isBipartite() 

{ return isTwoColorable; } 

} 


More examples of graph processing with DFS 
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routes.txt 

J FK MCO ' 
ORD DEN 
ORD HOU 
DFW PHX 


ATL MCO 
HOU MCO 
LAS PHX 


Symbol graphs Typical applications involve processing graphs defined in files or 
on web pages, using strings, not integer indices, to define and refer to vertices. To ac¬ 
commodate such applications, we define an input format with the following properties: 

■ Vertex names are strings. 

■ A specified delimiter separates vertex names (to allow for the possibility of 
spaces in names). 

■ Each line represents a set of edges, connecting the first vertex name on the line 
to each of the other vertices named on the line. 

■ The number of vertices V and the number of edges E are both implicitly defined. 
Shown below is a small example, the file routes. txt, which represents a model for a 
small transportation system where vertices are U.S. airport codes and edges connecting 
them are airline routes between the vertices. The file is simply a list of edges. Shown 

on the facing page is a larger example, taken from the 
file movies.txt, from the Internet Movie Database 
(IMDB), that we introduced in section 3 . 5 . Recall 
that this file consists of lines listing a movie name fol¬ 
lowed by a list of the performers in the movie. In the 
context of graph processing, we can view it as defining 
a graph with movies and performers as vertices and 
each line defining the adjacency list of edges connect¬ 
ing each movie to its performers. Note that the graph 
is a bipartite graph—there are no edges connecting 
performers to performers or movies to movies. 

API. The following API defines a Graph client that al¬ 
lows us to immediately use our graph-processing rou¬ 
tines for graphs defined by such files: 


— V and E 
not explicitly 
specified 



delimiter 


Symbol graph example (list of edges) 


public class SymbolGraph 


SymbolGraph(String filename, 

String delim) 

boolean contains(String key) 
int index(String key) 

String name(int v) 

Graph GO 


build graph specified in 
fi 1 ename using del im to 
separate vertex names 

is key a vertex? 
index associated with key 
key associated with index v 
underlying Graph 


API for graphs with symbolic vertex names 
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movies.txt 


V and E 
not explicitly 
— specified 


Tin Men (1987)/DeBoy, David/BIumenfeld, Alan/... /Ceppi, Cindy/Hershey, Barbara... 
Tirez sur 1e pianiste (1960)/Heymann, Claude/.../Berger, Nicole (I)... 

Titanic (1997)/Mazin, Stan/...DiCaprio, Leonardo/.../Winslet, Kate/... 

Titus (1999)/Wei sskopf, Hermann/Rhys, Matthew/. . ./McEwan , Geraldine "/"delimiter 
To Be or Not to Be (1942)/Verebes, Erno (I)/-../Lombard, Carole (I)... / 

To Be or Not to Be (1983)/.../Brooks, Mel (I)/.../Bancroft, Anne/... y' 

To Catch a Thief (1955)/Paris, Manuel/.../Grant, Cary/.../Kelly, Grace/... 

To Die For (1995)/Smith, Kurtwood/.../Kidman, Nicole/.../ Tucci, Maria... 


performers 


Symbol graph example (adjacency lists) 
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public static void main(String[] args) 

{ 

String filename = args[0]; 

String delim = args[l]; 

SymbolGraph sg = new SymbolGraph(filename, delim); 

Graph G = sg.GO ; 

while (Stdln.hasNextLine()) 

{ 

String source = Stdln . readLi ne() ; 
for (int w : G.adj(sg.index(source))) 
StdOut.println(" " + sg.name(w)); 

} 

} 

Test client for symbol graph API 


This API provides a construc¬ 
tor to read and build the graph 
and client methods nameO and 
i ndex() for translating vertex 
names between the strings on 
the input stream and the inte¬ 
ger indices used by our graph¬ 
processing methods. 

Test client. The test client at 
left builds a graph from the 
file named as the first com¬ 
mand-line argument (using 
the delimiter as specified by 


% java SymbolGraph routes.txt 
JFK 

ORD 

ATL 

MCO 

LAX 

LAS 

PHX 


% java SymbolGraph movies.txt "/" 
Tin Men (1987) 

DeBoy, David 
Blumenfeld, Alan 

Geppi, Cindy 
Hershey, Barbara 

Bacon, Kevin 

Mystic River (2003) 

Friday the 13th (1980) 
Flatliners (1990) 

Few Good Men, A (1992) 


the second command-line ar¬ 
gument) and then takes queries from standard input. 
The user specifies a vertex name and gets the list of 
vertices adjacent to that vertex. This client immedi¬ 
ately provides the useful inverted index functional¬ 
ity that we considered in section 3 . 5 . In the case of 
routes. txt, you can type an airport code to find the 
direct flights from that airport, information that is 
not directly available in the data file. In the case of 
movi es. txt, you can type the name of a performer 
to see the list of the movies in the database in which 
that performer appeared, or you can type the name 
of a movie to see the list of performers that appear 
in that movie. Typing a movie name and getting its 
cast is not much more than regurgitating the corre¬ 
sponding line in the input file, but typing the name 
of a performer and getting the list of movies in which 
that performer has appeared is inverting the index. 
Even though the database is built around connect¬ 
ing movies to performers, the bipartite graph model 
embraces the idea that it also connects performers 
to movies. The bipartite graph model automatically 
serves as an inverted index and also provides the basis 
for more sophisticated processing, as we will see. 
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this approach is clearly effective for any of the graph-processing methods that 
we consider: any client can use i ndex() when it wants to convert a vertex name to an 
index for use in graph processing and name() when it wants to convert an index from 
graph processing into a name for use in the context of the application. 

Implementation. A full Symbol Graph implementation is given on page 552. It builds 
three data structures: 

■ A symbol table st with Stri ng keys (vertex names) and i nt values (indices) 

■ An array keys [] that serves as an inverted index, giving the vertex name associ¬ 
ated with each integer index 

■ A Graph G built using the indices to refer to vertices 

Symbol Graph uses two passes through the data to build these data structures, pri¬ 
marily because the number of vertices V is needed to build the Graph. In typical real- 
world applications, keeping the value of V and E in the graph definition file (as in our 
Graph constructor at the beginning of this section) is somewhat inconvenient—with 
Symbol Graph, we can maintain files such as routes. txt or movi es. txt by adding or 
deleting entries withotit regard to the number of different names involved. 


symbol table 

ST<String, Integers st 


inverted index 

String[] keys 


undirected graph 

Graph G 




Symbol graph data structures 
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Symbol graph data type 


public class SymbolGraph 

{ 

private ST<String, Integer> st; 
private String[] keys; 
private Graph G; 


// String -> index 
// index -> String 
// the graph 


public SymbolGraphCString stream, String sp) 
{ 


st = new ST<String, Integer>(); 

In in = new In(stream); 
while (in. hasNextLineO) 

{ 

StringG a = in.readLine().spl 
for (int i = 0; i < a.length; 
if (!st.contains(a[i])) 
st.put(a[i], st.sizeO); 

} 

keys = new String[st.size()]; 
for (String name : st.keysO) 
keys[st.get(name)] = name; 

G = new Graph(st.sizeO) ; 
in = new In(stream); 
while (in.hasNextLineO) 

{ 

StringG a = in.readLine().spl 
int v = st.get(a[0]); 
for (int i = 1; i < a.length; 
G.addEdge(v, st.get(a[i])) 



// 

First pass 


// 

builds the index 

it(sp); 

// 

by reading strings 

i++) 

// 

to associate each 


// 

distinct string 


// 

with an index. 


// 

Inverted index 


// 

to get string keys 


// 

is an array. 


// 

Second pass 


// 

builds the graph 

it(sp); 

// 

by connecting the 


// 

fi rst vertex 

i++) 

// 

on each line 

J 

// 

to all the others. 


} 


} 


public boolean contains(String 
public int index(String s) 
public String name(int v) 
public Graph G() 


s) { return st.contains(s); } 

{ return st.get(s); } 

{ return keys[v]; } 

{ return G; } 


This Graph client allows clients to define graphs with Stri ng vertex names instead of integer indices. 
It maintains instance variables st (a symbol table that maps names to indices), keys (an array that 
maps indices to names), and G (a graph, with integer vertex names). To build these data structures, 
it makes two passes through the graph definition (each line has a string and a list of adjacent strings, 
separated by the delimiter sp). 


www.it-ebooks.info 






4.1 ■ Undirected Graphs 


553 


Degrees of separation. One of the classic applications of graph processing is to find 
the degree of separation between two individuals in a social network. To fix ideas, we 
discuss this application in terms of a recently popularized pastime known as the Kevin 
Bacon game, which uses the movie-performer graph that we just considered. Kevin Ba¬ 
con is a prolific actor who has appeared in many movies. We assign every performer a 
Kevin Bacon number as follows: Bacon himself is 0, any performer who has been in the 
same cast as Bacon has a Kevin Bacon number of 1, any other performer (except Bacon) 
who has been in the same cast as a performer whose number is 1 has a Kevin Bacon 
number of 2, and so forth. For example, Meryl Streep has a Kevin Bacon number of 1 
because she appeared in The River Wild with Kevin Bacon. Nicole Kidman’s number is 
2: although she did not appear in any movie with Kevin Bacon, she was in Days of Thun¬ 
der with Tom Cruise, and Cruise appeared in A Few Good Men with Kevin Bacon. Given 
the name of a performer, the simplest version of the game is to find some alternating se¬ 
quence of movies and performers that leads back to Kevin Bacon. For example, a movie 
buff might know that Tom Hanks was in Joe Versus the Volcano with Lloyd Bridges, 
who was in High Noon with 
Grace Kelly, who was in Dial 
M for Murder with Patrick 
Allen, who was in The Eagle 
Has Landed with Donald 
Sutherland, who was in 
Animal House with Kevin 
Bacon. But this knowledge 
does not suffice to establish 
Tom Hanks’s Bacon number 
(it is actually 1 because he 
was in Apollo 13 with Kevin 
Bacon). You can see that the 
Kevin Bacon number has to 
be defined by counting the movies in the shortest such sequence, so it is hard to be sure 
whether someone wins the game without using a computer. Of course, as illustrated in 
the Symbol Graph client Deg reesOfSe pa rati on onpage555,BreadthFi rst Paths is the 
program we need to find a shortest path that establishes the Kevin Bacon number of 
any performer in movi es. txt. This program takes a source vertex from the command 
line, then takes queries from standard input and prints a shortest path from the source 
to the query vertex. Since the graph associated with movi es . txt is bipartite, all paths 
alternate between movies and performers, and the printed path is a “proof” that the 
path is valid (but not a proof that it is the shortest such path—you need to educate your 


% java DegreesOfSeparation movies.txt "/" "Bacon, Kevin" 
Kidman, Nicole 
Bacon, Kevin 
Few Good Men, A (1992) 

Cruise, Tom 

Days of Thunder (1990) 

Kidman, Nicole 
Grant, Cary 
Bacon, Kevin 
Mystic River (200B) 

Willis, Susan 
Majestic, The (2001) 

Landau, Martin 

North by Northwest (1959) 

Grant, Cary 
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friends about proposition b for that). DegreesOfSeparati on also finds shortest paths 
in graphs that are not bipartite: for example, it finds a way to get from one airport to 
another in routes . txt using the fewest connections. 

you might enjoy using DegreesOfSeparati on to answer some entertaining ques¬ 
tions about the movie business. For example, you can find separations between mov¬ 
ies, not just performers. More important, the concept of separation has been widely 
studied in many other contexts. For example, mathematicians play this same game with 
the graph defined by paper co-authorship and their connection to P. Erdos, a prolific 
20th-century mathematician. Similarly, everyone in New Jersey seems to have a Bruce 
Springsteen number of 2, because everyone in the state seems to know someone who 
claims to know Bruce. To play the Erdos game, you would need a database of all math¬ 
ematical papers; playing the Springsteen game is a bit more challenging. On a more 
serious note, degrees of separation play a crucial role in the design of computer and 
communications networks, and in our understanding of natural networks in all fields 
of science. 


% java DegreesOfSeparati on movies.txt "/" "Animal House (1978)" 
Titanic (1997) 

Animal House (1978) 

Allen, Karen (I) 

Raiders of the Lost Ark (1981) 

Taylor, Rocky (I) 

Titanic (1997) 

To Catch a Thief (1955) 

Animal House (1978) 

Vernon, John (I) 

Topaz (1969) 

Hitchcock, Alfred (I) 

To Catch a Thief (1955) 
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Degrees of separation 


public class DegreesOfSeparation 

{ 

public static void main(String[] args) 

{ 

SymbolGraph sg = new SymbolGraph(args[0], args[l]); 

Graph G = sg .GO ; 

String source = args[2]; 
if (! sg.contains(source)) 

{ StdOut.printlnfsource + " not in database."); return; } 
int s = sg.index(source); 

BreadthFirstPaths bfs = new BreadthFirstPaths(G, s); 

while (!Stdln.isEmptyO) 

{ 

String sink = Stdln.readLine(); 
if (sg.contains(sink)) 

{ 

int t = sg.indexfsink); 
if (bfs.hasPathTo(t)) 

for (int v : bfs.pathTo(t)) 

StdOut.println(" " + sg.name(v)); 
else StdOut.println("Not connected"); 

} 

else StdOut.println("Not in database."); 

} 

} 

} 


This SymbolGraph and BreadthFi rstPaths client finds shortest paths in graphs. For movies.txt, it 
plays the Kevin Bacon game. 

% java DegreesOfSeparation routes.txt " " JFK 
LAS 

JFK 

ORD 

PHX 

LAS 

DFW 

JFK 

ORD 

DFW 


www.it-ebooks.info 






556 


CHAPTER 4 ■ Graphs 


Summary In this section, we have introduced several basic concepts that we will 
expand upon and further develop throughout the rest of this chapter: 

■ Graph nomenclature 

■ A graph representation that enables processing of huge sparse graphs 

■ A design pattern for graph processing, where we implement algorithms by devel¬ 
oping clients that preprocess the graph in the constructor, building data struc¬ 
tures that can efficiently support client queries about the graph 

■ Depth-first search and breadth-first search 

■ A class providing the capability to use symbolic vertex names 

The table below summarizes the implementations of graph algorithms that we have 
considered. These algorithms are a proper introduction to graph processing, since vari¬ 
ants on their code will resurface as we consider more complicated types of graphs and 
applications, and (consequently) more difficult graph-processing problems. The same 
questions involving connections and paths among vertices become much more difficult 
when we add direction and then weights to graph edges, but the same approaches are 
effective in addressing them and serve as a starting point for addressing more difficult 
problems. 


problem 

solution 

reference 

single-source connectivity 

DepthFirstSearch 

page 531 

single-source paths 

DepthFirstPaths 

page 536 

single-source shortest paths 

BreadthFirstPaths 

page 540 

connectivity 

CC 

page 544 

cycle detection 

Cycle 

page 547 

two-colorability (bipartiteness) 

TwoColor 

page 547 


(Undirected) graph-processing problems addressed in this section 
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Q&A 


Q. Why not jam all of the algorithms into Graph .java? 

A. Yes, we might just add query methods (and whatever private fields and methods 
each might need) to the basic Graph ADT definition. While this approach has some of 
the virtues of data abstraction that we have embraced, it also has some serious draw¬ 
backs, because the world of graph processing is significantly more expansive than the 
kinds of basic data structures treated in section 1 . 3 . Chief among these drawbacks are 
the following: 

■ There are many more graph-processing operations to implement than we can 
accurately define in a single API. 

■ Simple graph-processing tasks have to use the same API needed by complicated 
tasks. 

■ One method can access a field intended for use by another method, contrary to 
encapsulation principles that we would like to follow. 

This situation is not unusual: APIs of this kind have come to be known as wide inter¬ 
faces (see page 97). In a chapter filled with graph-processing algorithms, an API of this 
sort would be wide indeed. 

Q. Does Symbol Graph really need two passes? 

A. No. You could pay an extra lg N factor and support ad j () directly as an ST instead 
of a Bag. We have an implementation along these lines in our book An Introduction to 
Programming in Java: An Interdisciplinary Approach. 
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EXERCISES 

4.1.1 What is the maximum number of edges in a graph with V vertices and no paral¬ 
lel edges? What is the minimum number of edges in a graph with V vertices, none of 
which are isolated? 


4 . 1.2 Draw, in the style of the figure in the text (page 524), the ad¬ 
jacency lists built by Graph’s input stream constructor for the file 
ti nyGex2 . txt depicted at left. 

4 . 1.3 Create a copy constructor for Graph that takes as input a 
graph G and creates and initializes a new copy of the graph. Any 
changes a client makes to G should not affect the newly created 
graph. 

4 . 1.4 Add a method hasEdgeO to Graph which takes two int 
arguments v and w and returns true if the graph has an edge v-w, 
fal se otherwise. 

4 . 1.5 Modify Graph to disallow parallel edges and self-loops. 

4 . 1.6 Consider the four-vertex graph with edges 0 -1, 1-2, 2-3, 
and 3-0. Draw an array of adjacency-lists that could not have been 

built calling addEdge () for these edges no matter what order. 

4 . 1.7 Develop a test client for Graph that reads a graph from the input stream named 
as command-line argument and then prints it, relying on toStri ng(). 

4 . 1.8 Develop an implementation for the Search API on page 528 that uses UF, as de¬ 
scribed in the text. 

4 . 1.9 Show, in the style of the figure on page 533, a detailed trace of the call df s (0) for 
the graph built by Graph’s input stream constructor for the file ti nyGex2 . txt (see ex¬ 
ercise 4 . 1 . 2 ). Also, draw the tree represented by edgeTo [] . 

4 . 1.10 Prove that every connected graph has a vertex whose removal (including all 
adjacent edges) will not disconnect the graph, and write a DFS method that finds such 
a vertex. Hint : Consider a vertex whose adjacent vertices are all marked. 

4 . 1.11 Draw the tree represented by edgeTo [] after the call bfs (G , 0) in algorithm 
4.2 for the graph built by Graph’s input stream constructor for the file ti nyGex2 . txt 
(see exercise 4 . 1 . 2 ). 


tinyGex2.txt 
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4 . 1.12 What does the BFS tree tell us about the 
distance from v to w when neither is at the root? 

4 . 1.13 Add a distToO method to the 
BreadthFi rstPaths API and implementation, 
which returns the number of edges on the shortest 
path from the source to a given vertex. AdistToO 
query should run in constant time. 

4 . 1.14 Suppose you use a stack instead of a queue 
when running breadth-first search. Does it still 
compute shortest paths? 

4 . 1.15 Modify the input stream constructor for 
Graph to also allow adjacency lists from standard 
input (in a manner similar to Symbol Graph), as in 
the example ti nyGadj . txt shown at right. After 
the number of vertices and edges, each line con¬ 
tains a vertex and its list of adjacent vertices. 




same lists as for 
list-of-edges input 
but order within lists 
is different 

/ 


tinyGadj.txt 



0 12 5 6 


3 4 5 

4 5 6 
7 8 

9 10 11 12 
11 12 


% java Graph tinyGadj.txt 
13 vertices, 13 edges 


0: 6 5 2 1 
1:0 y 

4: 6 5 3 / 
5: 4 3 0 / 
6: 4 0 / 

7 : 8 / 

8:7 t 
9: 12 11 10 
10: 9 
11: 12 9 
12: 11 9 


list order 
r is reversed 
from input 


second 

representation 
of each edge 
appears in red 


4 . 1.16 The eccentricity of a vertex v is the the length of the shortest path from that ver¬ 
tex to the furthest vertex from v. The diameter of a graph is the maximum eccentricity 
of any vertex. The radius of a graph is the smallest eccentricity of any vertex. A center is 
a vertex whose eccentricity is the radius. Implement the following API: 


public class GraphProperties 


GraphProperties(Graph G) 
int eccentricity(int v) 
int diameterO 
int radiusO 
int centerQ 


constructor (exception if G not connected) 
eccentricity of\/ 
diameter of G 
radius of G 
a center of G 


4.1.18 The girth of a graph is the length of its shortest cycle. If a graph is acyclic, then its 
girth is infinite. Add a method gi rth() to GraphProperti es that returns the girth of 
the graph. Hint: Run BFS from each vertex. The shortest cycle containing s is a shortest 
path from s to some vertex v, plus the edge from v back to s. 


www.it-ebooks.info 







560 


CHAPTER 4 ■ Graphs 


EXERCISES (continued) 


4 . 1.19 Show, in the style of the figure on page 545, a detailed trace of CC for finding the 
connected components in the graph built by Graph’s input stream constructor for the 
file ti nyGex2 . txt (see exercise 4 . 1 . 2 ). 

4 . 1.20 Show, in the style of the figures in this section, a detailed trace of Cycle for 
finding a cycle in the graph built by Graph’s input stream constructor for the file 
ti nyGex2 . txt (see exercise 4 . 1 . 2 ). What is the order of growth of the running time 
of the Cycl e constructor, in the worst case? 

4 . 1.21 Show, in the style of the figures in this section, a detailed trace of TwoCol or for 
finding a two-coloring of the graph built by Graph’s input stream constructor for the 
file tinyGex2.txt (see exercise 4 . 1 . 2 ). What is the order of growth of the running 
time of the TwoCol or constructor, in the worst case? 

4 . 1.22 Run Symbol Graph with movies.txt to find the Kevin Bacon number of this 
year’s Oscar nominees. 

4 . 1.23 Write a program BaconHi stogram that prints a histogram of Kevin Bacon 
numbers, indicating how many performers from movi es. txt have a Bacon number of 
0, 1,2, 3,.... Include a category for those who have an infinite number (not connected 
to Kevin Bacon). 

4 . 1.24 Compute the number of connected components in movi es. txt, the size of the 
largest component, and the number of components of size less than 10. Find the eccen¬ 
tricity, diameter, radius, a center, and the girth of the largest component in the graph. 
Does it contain Kevin Bacon? 

4 . 1.25 Modify DegreesOfSeparati on to take an i nt value y as a command-line argu¬ 
ment and ignore movies that are more than y years old. 

4 . 1.26 Write a SymbolGraph client like DegreesOfSeparati on that uses depth-first 
search instead of breadth-first search to find paths connecting two performers, produc¬ 
ing output like that shown on the facing page. 
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4 . 1.27 Determine the amount of memory used by Graph to represent a graph with V 
vertices and E edges, using the memory-cost model of section 1 . 4 . 

4 . 1.28 Two graphs are isomorphic if there is a way to rename the vertices of one to make 
it identical to the other. Draw all the nonisomorphic graphs with two, three, four, and 
five vertices. 

4 . 1.29 Modify Cycl e so that it works even if the graph contains self-loops and parallel 
edges. 


% java DegreesOfSeparationDFS movies.txt 
Source: Bacon, Kevin 
Query: Kidman, Nicole 
Bacon, Kevin 
Mystic River (200B) 

O’Hara, Jenny 
Matchstick Men (200B) 

Grant, Beth 
... [123 movies ] (!) 

Law, Jude 

Sky Captain... (2004) 

Jolie, Angelina 
Playing by Heart (1998) 

Anderson, Gillian (I) 

Cock and Bull Story, A (2005) 
Henderson, Shirley (I) 

24 Hour Party People (2002) 

Eccleston, Christopher 
Cone in Sixty Seconds (2000) 
Balahoutis, Alexandra 
Days of Thunder (1990) 

Kidman, Nicole 
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| CREATIVE PROBLEMS 

4 . 1.30 Eulerian and Hamiltonian cycles. Consider the graphs defined by the following 
four sets of edges: 

0-1 0-2 0-3 1-3 1-4 2-5 2-9 3-6 4-7 4-8 5-8 5-9 6-7 6-9 7-8 

0-1 0-2 0-3 1-3 0-3 2-5 5-6 3-6 4-7 4-8 5-8 5-9 6-7 6-9 8-8 

0-1 1-2 1-3 0-3 0-4 2-5 2-9 3-6 4-7 4-8 5-8 5-9 6-7 6-9 7-8 

4-1 7-9 6-2 7-3 5-0 0-2 0-8 1-6 3-9 6-3 2-8 1-5 9-8 4-5 4-7 

Which of these graphs have Euler cycles (cycles that visit each edge exactly once)? 
Which of them have Hamilton cycles (cycles that visit each vertex exactly once)? 

4 . 1.31 Graph enumeration. How many different undirected graphs are there with V 
vertices and E edges (and no parallel edges)? 

4 . 1.32 Parallel edge detection. Devise a linear-time algorithm to count the parallel 
edges in a graph. 

4 . 1.33 Odd cycles. Prove that a graph is two-colorable (bipartite) if and only if it con¬ 
tains no odd-length cycle. 

4 . 1.34 Symbol graph. Implement a one-pass Symbol Graph (it need not be a Graph 
client). Your implementation may pay an extra log V” factor for graph operations, for 
symbol-table lookups. 

4 . 1.35 Biconnectedness. A graph is biconnected if every pair of vertices is connected 
by two disjoint paths. An articulation point in a connected graph is a vertex that would 
disconnect the graph if it (and its adjacent edges) were removed. Prove that any graph 
with no articulation points is biconnected. Hint: Given a pair of vertices s and t and a 
path connecting them, use the fact that none of the vertices on the path are articulation 
points to construct two disjoint paths connecting s and t. 

4 . 1.36 Edge connectivity. A bridge in a graph is an edge that, if removed, would sepa¬ 
rate a connected graph into two disjoint subgraphs. A graph that has no bridges is said 
to be edge connected. Develop a DFS-based data type for determing whether a given 
graph is edge connected. 

4 . 1.37 Euclidean graphs. Design and implement an API Eucl i deanGraph for graphs 
whose vertices are points in the plane that include coordinates. Include a method 
showQ that uses StdDraw to draw the graph. 
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4.1.38 Image processing. Implement the flood fill operation on the implicit graph de¬ 
fined by connecting adjacent points that have the same color in an image. 
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EXPERIMENTS 

4.1.39 Random graphs. Write a program ErdosRenyiGraph that takes integer values 
V and E from the command line and builds a graph by generating E random pairs of in¬ 
tegers between 0 and V — 1. Note: This generator produces self-loops and parallel edges. 

4.1.40 Random simple graphs. Write a program RandomSimpl eGraph that takes inte¬ 
ger values V and E from the command line and produces, with equal likelihood, each of 
the possible simple graphs with V vertices and E edges. 

4.1.41 Random sparse graphs. Write a program RandomSparseGraph to generate ran¬ 
dom sparse graphs for a well-chosen set of values of V and E such that you can use it to 
run meaningful empirical tests on graphs drawn from the Erdos-Renyi model. 

4.1.42 Random Euclidean graphs. Write a EuclideanGraph client (see exercise 
4 . 1 . 37 ) RandomEucl i deanGraph that produces random graphs by generating V random 
points in the plane, then connecting each point with all points that are within a circle 
of radius d centered at that point. N ote: The gr aph will almost certainly be connected if 
d is larger than the threshold value y 1 g V/tt V and almost certainly disconnected if d is 
smaller than that value. 

4.1.43 Random grid graphs. Write a Eucl ideanGraph client RandomGridGraph that 
generates random graphs by connecting vertices arranged in a Jv-by-Jv grid to their 
neighbors (see exercise 1 . 5 . 15 ). Augment your program to add R extra random edges. 
For large R, shrink the grid so that the total number of edges remains about V. Add 
an option such that an extra edge goes from a vertex s to a vertex t with probability 
inversely proportional to the Euclidean distance between s and t. 

4.1.44 Real-world graphs. Find a large weighted graph on the web—perhaps a map 
with distances, telephone connections with costs, or an airline rate schedule. Write a 
program RandomReal Graph that builds a graph by choosing V vertices at random and 
E edges at random from the subgraph induced by those vertices. 

4.1.45 Random interval graphs. Consider a collection of V intervals on the real line 
(pairs of real numbers). Such a collection defines an interval graph with one vertex cor¬ 
responding to each interval, with edges between vertices if the corresponding intervals 
intersect (have any points in common). Write a program that generates V random in¬ 
tervals in the unit interval, all of length d, then builds the corresponding interval graph. 
Hint: Use a BST. 
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4.1.46 Random transportation graphs. One way to define a transportation system is 
with a set of sequences of vertices, each sequence defining a path connecting the ver¬ 
tices. For example, the sequence 0-9-3-2 defines the edges 0-9, 9-3, and 3-2. Write a 
Eucl i deanGraph client RandomTransportati on that builds a graph from an input file 
consisting of one sequence per line, using symbolic names. Develop input suitable to al¬ 
low you to use your program to build a graph corresponding to the Paris Metro system. 

Testing all algorithms and studying all parameters against all graph models is unrealistic. 
For each problem listed below, write a client that addresses the problem for any given input 
graph, then choose among the generators above to run experiments for that graph model. 
Use your judgment in selecting experiments, perhaps in response to results of previous 
experiments. Write a narrative explaining your results and any conclusions that might be 
drawn. 


4.1.47 Path lengths in DFS. Run experiments to determine empirically the probability 
that DepthFi rstPaths finds a path between two randomly chosen vertices and to cal¬ 
culate the average length of the paths found, for various graph models. 

4.1.48 Path lengths in BFS. Run experiments to determine empirically the probability 
that BreadthFi rstPaths finds a path between two randomly chosen vertices and to 
calculate the average length of the paths found, for various graph models. 

4.1.49 Connected components. Run experiments to determine empirically the distri¬ 
bution of the number of components in random graphs of various types, by generating 
large numbers of graphs and drawing a histogram. 

4.1.50 Two-colorable. Most graphs are not two-colorable, and DFS tends to discov¬ 
er that fact quickly. Run empirical tests to study the number of edges examined by 
TwoCol or, for various graph models. 
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4.2 DIRECTED GRAPHS 


In directed graphs, edges are one-way: the pair of vertices that defines each edge is 
an ordered pair that specifies a one-way adjacency. Many applications (for example, 
graphs that represent the web, scheduling constraints, or telephone calls) are naturally 

expressed in terms of directed graphs. The 
one-way restriction is natural, easy to en¬ 
force in our implementations, and seems in¬ 
nocuous; but it implies added combinatori¬ 
al structure that has profound implications 
for our algorithms and makes working with 
directed graphs quite different from work¬ 
ing with undirected graphs. In this section, 
we consider classic algorithms for exploring 
and processing directed graphs. 


application 

vertex 

edge 

food web 

species 

predator-prey 

internet content 

page 

hyperlink 

program 

module 

external reference 

cellphone 

phone 

call 

scholarship 

paper 

citation 

financial 

stock 

transaction 

internet 

machine 

connection 


Typical digraph applications 


Glossary Our definitions for directed 
graphs are nearly identical to those for un¬ 
directed graphs (as are some of the algo¬ 
rithms and programs that we use), but they 
are worth restating. The slight differences in the wording to account for edge directions 
imply structural properties that will be the focus of this section. 


Definition. A directed graph (or digraph) is a set of vertices and a collection of di¬ 
rected edges. Each directed edge connects an ordered pair of vertices. 


We say that a directed edge points from the first vertex in the pair and points to the 
second vertex in the pair. The outdegree of a vertex in a digraph is the number of edges 
going from it; the indegree of a vertex is the number of edges going to it. We drop the 
modifier directed when referring to edges in digraphs when the distinction is obvious in 
context. The first vertex in a directed edge is called its head ; the second vertex is called 
its tail. We draw directed edges as arrows pointing from head to tail. We use the nota¬ 
tion v->w to refer to an edge that points from v to w in a digraph. As with undirected 
graphs, our code handles parallel edges and self-loops, but they are not present in ex¬ 
amples and we generally ignore them in the text. Ignoring anomalies, there are four 
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different ways in which two vertices might be related in a digraph: no edge; an edge 
v->w from v to w; an edge w->v from w to v; or two edges v->w and w->v, which indicate 
connections in both directions. 


Definition. A directed path in a digraph is a sequence of vertices in which there is 
a (directed) edge pointing from each vertex in the sequence to its successor in the 
sequence. A directed cycle is a directed path with at least one edge whose first and 
last vertices are the same. A simple cycle is a cycle with no repeated edges or vertices 
(except the requisite repetition of the first and last vertices). The length of a path or 
a cycle is its number of edges. 


As for undirected graphs, we assume that directed paths 
are simple unless we specifically relax this assumption by 
referring to specific repeated vertices (as in our definition 
of directed cycle) or to general directed paths. We say that 
a vertex w is reachable from a vertex v if there is a directed 
path from v to w. Also, we adopt the convention that each 
vertex is reachable from itself. Except for this case, the fact 
that w is reachable from v in a digraph indicates nothing 
about whether v is reachable from w. This distinction is 
obvious, but critical, as we shall see. 

understanding the algorithms in this section requires an appreciation of the dis¬ 
tinction between reachability in digraphs and connectivity in undirected graphs. De¬ 
veloping such an appreciation is more complicated than you might think. For example, 

although you are likely to be able to tell at a glance 
whether two vertices in a small undirected graph are 
connected, a directed path in a digraph is not so easy 
to spot, as indicated in the example at left. Processing 
digraphs is akin to traveling around in a city where 
all the streets are one-way, with the directions not 
necessarily assigned in any uniform pattern. Getting 
from one point to another in such a situation could 
be a challenge indeed. Counter to this intuition is 
the fact that the standard data structure that we use 
for representing digraphs is simpler than the corre¬ 
sponding representation for undirected graphs! 



Is w reachable from v in this digraph? 


directed 
cycle of 
length 3 


vertex of 
indegree 3 and 
outdegree 2 


directed 

edge 



directed 
path of 
length 4 


Anatomy of a digraph 
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Digraph data type The API below and the class Digraph shown on the facing 
page are virtually identical to those for Graph (page 526). 


public class Digraph 


Digraph(int V) 
Digraph(In in) 
int VO 
int EO 

void addEdge(int v, int w) 


Iterable<Integer> adjCint v) 


Digraph reversef) 
String toStringO 


create a V-vertex digraph with no edges 

read a digraph from input stream i n 

number of vertices 

number of edges 

add edge v->w to this digraph 

vertices connected to v by edges 
pointing from v 

reverse of this digraph 
string representation 


API for a digraph 


Representation. We use the adjacency-lists representation, where an edge v->w is rep¬ 
resented as a list node containing w in the linked list corresponding to v. This represen¬ 
tation is essentially the same as for undirected graphs but is even more straightforward 
because each edge occurs just once, as shown on the facing page. 

Input format. The code for the constructor that takes a digraph from an input stream 
is identical to the corresponding constructor in Graph—the input format is the same, 
but all edges are interpreted to be directed edges. In the list-of-edges format, a pair v w 
is interpreted as an edge v->w. 

Reversing a digraph. Digraph also adds to the API a method reverseO which re¬ 
turns a copy of the digraph, with all edges reversed. This method is sometimes needed 
in digraph processing because it allows clients to find the edges that point to each ver¬ 
tex, while ad j () gives just vertices connected by edges that point from each vertex. 

Symbolic names. It is also a simple matter to allow clients to use symbolic names in 
digraph applications. To implement a class Symbol Digraph like Symbol Graph on page 
552, replace Graph by Digraph everywhere. 

it is worthwhile to take the time to consider carefully the difference, by comparing 
code and the figure at right with their counterparts for undirected graphs on page 524 
and page 526. In the adjacency-lists representation of an undirected graph, we know that 
if v is on w’s list, then w will be on v’s list; the adjacency-lists representation of a di¬ 
graph has no such symmetry. This difference has profound implications in processing 
digraphs. 
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Directed graph (digraph) data type 


public class Digraph 

{ 

private final int V; 

private int E; 

private Bag<Integer>[] adj ; 

public Digraph(int V) 

{ 

this.V = V; 
this.E = 0; 

adj = (Bag<Integer>[]) new Bag[V]; 
for (int v = 0; v < V; v++) 
adj[v] = new Bag<Integer>() ; 

} 

public int V() { return V; } 
public int E() { return E; } 

public void addEdge(int v, int w) 

{ 

adj[v].add(w); 

E++; 

} 

public Iterable<Integer> adj(int v) 

{ return adj[v]; } 

public Digraph reverseO 

{ 

Digraph R = new Digraph(V); 
for (int v = 0; v < V; v++) 
for (int w : adj(v)) 
R.addEdge(w, v); 
return R; 

} 


This Di graph data type is identical to Graph (page 526) ex¬ 
cept that add Edge O only calls add() once, and it has an 
instance method reverseO that returns a copy with all 
its edges reversed. Since the code is easily derived from the 
corresponding code for Graph, we omit the toStringO 
method (see the table on page 523) and the input stream con¬ 
structor from (see page 526). 


tinyDG.txt 

^ 13 

22 

4 2 

2 3 



11 

4 
3 

7 

8 

5 
0 



Digraph input format and 
adjacency-lists representation 
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Reachability in digraphs Our first graph-processing algorithm for undirected 
graphs was Depth Fi rstSearch on page 531, which solves the single-source connectivity 
problem, allowing clients to determine which vertices are connected to a given source. 
The identical code with Graph changed to Digraph solves the analogous problem for 
digraphs: 

Single-source reachability. Given a digraph and a source vertex s, support que¬ 
ries of the form Is there a directed path from s to a given target vertex v? 

Di rectedDFS on the facing page is a slight embellishment of DepthFi rstSearch that 
implements the following API: 

public class Di rectedDFS 

DirectedDFS(Digraph G, int s) 

DirectedDFS(Digraph G, 

Iterable<Integer> sources) 

boolean marked(int v) 

API for reachability in digraphs 

By adding a second constructor that takes a list of vertices, this API supports for clients 
the following generalization of the problem: 

Multiple-source reachability. Given a digraph and a set of source vertices, sup¬ 
port queries of the form Is there a directed path from any vertex in the set to a given 
target vertex v? 

This problem arises in the solution of a classic string-processing problem that we con¬ 
sider in section 5 . 4 . 

Di rectedDFS uses our standard graph-processing paradigm and a standard recur¬ 
sive depth-first search to solve these problems. It calls the recursive dfs() for each 
source, which marks every vertex encountered. 


find vertices in G that are 
reachable from s 

find vertices in G that are 
reachable from sources 

is v reachable? 


Proposition D. DFS marks all the vertices in a digraph reachable from a given set of 
sources in time proportional to the sum of the outdegrees of the vertices marked. 

Proof: Same as proposition a on page 531. 

A trace of the operation of this algorithm for our sample digraph appears on page 572. 
This trace is somewhat simpler than the corresponding trace for undirected graphs, 
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ALGORITHM 4.4 Reachability in digraphs 


public class DirectedDFS 

{ 

private boolean[] marked; 

public DirectedDFS(Digraph G, int s) 

{ 

marked = new boolean[G.V()]; 
dfs(G, s); 

} 

public DirectedDFS(Digraph G, Iterable<Integer> sources) 

{ 

marked = new boolean[G.V()]; 
for (int s : sources) 

if (!marked[s]) dfs(G, s); 

} 

private void dfs(Digraph G, int v) 

{ 

marked[v] = true; 
for (int w : G.adj(v)) 

if (!marked[w]) dfs(G, w); 

} 

public boolean marked(int v) 

{ return marked[v]; } 

public static void main(String[] args) 

{ 

Digraph G = new Digraph(new ln(args[0])); 

Bag<Integer> sources = new Bag<Integer>(); 
for (int i =1; i < args.length; i++) 

sources.add(Integer.parselnt(args[i])); 

DirectedDFS reachable = new DirectedDFS(G, sources); 

for (int v = 0; v < G.V(); v++) 

if (reachable.marked(v)) StdOut.print(v + " "); 
StdOut.printlnQ ; 


% java DirectedDFS 
1 

% java DirectedDFS 
0 1 2 3 4 5 

% java DirectedDFS 
01234569 10 


tinyDG.txt 1 

tinyDG.txt 2 

tinyDG.txt 126 
11 12 


This implementation of depth-first search provides clients the ability to test which vertices are reach¬ 
able from a given vertex or a given set of vertices. 
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dfs(O) 


dfs(5) 


dfs(4) 


dfs(3) 
check 5 


dfs(2) 
check 0 
check 3 
2 done 
3 done 
check 2 
4 done 
5 done 


dfs(l) 
1 done 
0 done 





marked [] 
0 T 
1 
2 

3 

4 

5 


0 T 

1 

2 

3 

4 

5 T 


0 T 

1 

2 

3 

4 T 

5 T 


0 T 

1 

2 

3 T 

4 T 

5 T 


0 T 
1 

2 T 

3 T 

4 T 

5 T 


0 T 

1 T 

2 T 

3 T 

4 T 

5 T 


ad j [] 

0 5 1 

1 

2 0 3 

3 5 2 

4 3 2 

5 4 


0 5 1 

1 

2 0 3 

3 5 2 

4 3 2 

5 4 


0 5 1 

1 

2 0 3 

3 5 2 

4 3 2 

5 4 


0 5 1 

1 

2 0 3 

3 5 2 

4 3 2 

5 4 


0 5 1 

1 

2 0 3 

3 5 2 

4 2 

5 4 


0 5 1 

1 
2 

3 5 2 

4 

5 4 


Trace of depth-first search to find vertices reachable from vertex 0 in a digraph 
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because DFS is fundamentally a digraph-processing algorithm, with one representation 
of each edge. Following this trace is a worthwhile way to help cement your understand¬ 
ing of depth-first search in digraphs. 

Mark-and-sweep garbage collection. An im¬ 
portant application of multiple-source reach¬ 
ability is found in typical memory-management 
systems, including many implementations of 
Java. A digraph where each vertex represents 
an object and each edge represents a reference 
to an object is an appropriate model for the 
memory usage of a running Java program. At 
any point in the execution of a program, certain 
objects are known to be directly accessible, and 
any object not reachable from that set of objects 
can be returned to available memory. A mark- 
and-sweep garbage collection strategy reserves 
one bit per object for the purpose of garbage 
collection, then periodically marks the set of 
potentially accessible objects by running a di¬ 
graph reachability algorithm like Di rectedDFS 
and sweeps through all objects, collecting the 
unmarked ones for use for new objects. 

Finding paths in digraphs. DepthFi rstPaths (algorithm 4.1 on page 536) and 
BreadthFi rstPaths (algorithm 4.2 on page 540) are also fundamentally digraph¬ 
processing algorithms. Again, the identical APIs and code (with Graph changed to 
Digraph) effectively solve the following problems: 

Single-source directed paths. Given a digraph and a source vertex s, support 
queries of the form Is there a directed path from s to a given target vertex v? If so, 
find such a path. 

Single-source shortest directed paths. Given a digraph and a source vertex s, 
support queries of the form Is there a directed path from s to a given target vertex 
v? If so, find a shortest such path (one with a minimal number of edges). 

On the booksite and in the exercises at the end of this section, we refer to these solu¬ 
tions as DepthFi rstDi rectedPaths and BreadthFi rstDi rectedPaths, respectively. 


directly 

accessible 



objects 

Garbage collection scenario 
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Cycles and DAGs Directed cycles are of partic¬ 
ular importance in applications that involve process¬ 
ing digraphs. Identifying directed cycles in a typical 
digraph can be a challenge without the help of a 
computer, as shown at right. In principle, a digraph 
might have a huge number of cycles; in practice, we 
typically focus on a small number of them, or simply 
are interested in knowing that none are present. 

To motivate the study of the role of directed cycles 
in digraph processing we consider, as a running ex¬ 
ample, the following prototypical application where 
digraph models arise directly: 

Scheduling problems. A widely applicable problem-solving model has to do with ar¬ 
ranging for the completion of a set of jobs, under a set of constraints, by specifying 
when and how the jobs are to be performed. Constraints might involve functions of 
the time taken or other resources consumed by the jobs. The most important type 
of constraints is precedence constraints, which specify that certain tasks must be per¬ 
formed before certain others. Different types of additional constraints lead to many 
different types of scheduling problems, of varying difficulty. Literally thousands of dif¬ 
ferent problems have been studied, and researchers still seek better algorithms for many 
of them. As an example, consider a college student planning a course schedule, under 
the constraint that certain courses are prerequisite for certain other courses, as in the 
example below. 



Does this digraph have a directed cycle? 



( Databases ) ( Introduction to CS) 


Linear Algebra ) -<— ( Calculus ) 

( Theoretical CS 

\ 

( Artificial Intelligence )-Robotics ) 


( Advanced Programming j 

^- 7 -^—-,, C Machine Learning )- -C Neural Networks ; 

Q Computational Biology ) v - —' '- ' 


( Scientific Computing 


A precedence-constrained scheduling problem 
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If we further assume that the student can take only one course at a time, we have an 
instance of the following problem: 

Precedence-constrained scheduling. Given a set of jobs to be completed, with 
precedence constraints that specify that certain jobs have to be completed before 
certain other jobs are begun, how can we schedule the jobs such that they are all 
completed while still respecting the constraints? 



For any such problem, a digraph model is immediate, with 
vertices corresponding to jobs and directed edges cor¬ 
responding to precedence constraints. For economy, we 
switch the example to our standard model with vertices 
labeled as integers, as shown at left. In digraphs, prece¬ 
dence-constrained schedul¬ 
ing amounts to the following 
fundamental problem: 

Topological sort. Given a digraph, put the vertices in 
order such that all its directed edges point from a ver¬ 
tex earlier in the order to a vertex later in the order (or 
report that doing so is not possible). 


Standard digraph model 


edges all 
point down 


A topological order for our example model is shown at 
right. All edges point down, so it clearly represents a so¬ 
lution to the precedence-constrained scheduling prob¬ 
lem that this digraph models: the student can satisfy all 
course prerequisites by taking the courses in this order. 
This application is typical—some other representative 
applications are listed in the table below. 


application 

vertex 

edge 

job schedule 

job 

precedence constraint 

course schedule 

course 

prerequisite 

inheritance 

Java class 

extends 

spreadsheet 

cell 

formula 

symbolic links 

file name 

link 



prerequisites 
all satisfied 


Calculus 


Linear Algebra 


Intro 


to CS 


Advanced Programming 


Algorit 


Theoretical C 


Artificial Intelligence 


Robotics 


Machine Learning 


Neural Networks 


Databases 


Scienti 


mputing 


Computational Biology 


Typical topological-sort applications 


Topological sort 
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Cycles in digraphs. If job x must be completed before job y, job y before job z, and 
job z before job x, then someone has made a mistake, because those three constraints 
cannot all be satisfied. In general, if a precedence-constrained scheduling problem has 
a directed cycle, then there is no feasible solution. To check for such errors, we need to 
be able to solve the following problem: 

Directed cycle detection. Does a given digraph have a directed cycle? If so, find 
the vertices on some such cycle, in order from some vertex back to itself. 

A graph may have an exponential number of cycles (see exercise 4 . 2 . 11 ) so we only ask 
for one cycle, not all of them. For job scheduling and many other applications it is re¬ 
quired that no directed cycle exists, so digraphs where they are absent play a special role: 

Definition. A directed acyclic graph (DAG) is a digraph with no directed cycles. 


Solving the directed cycle detection problem thus answers the following question: Is a 
given digraph a DAG ? Developing a depth-first-search-based solution to this problem 
is not difficult, based on the fact that the recursive call stack maintained by the system 
represents the “current” directed path under consideration (like the string back to the 
entrance in Tremaux maze exporation). If we ever find a directed edge v->w to a vertex 
w that is on that stack, we have found a cycle, since the stack is evidence of a directed 
path from w to v, and the edge v->w completes the cycle. Moreover, the absence of any 
such back edges implies that the graph is acyclic. Di rectedCycle on the facing page 
uses this idea to implement the following API: 

public class Di rectedCycle 

DirectedCycle(Digraph G) 
boolean hasCycleO 
Iterable<Integer> cycleO 

API for reachability in digraphs 


cycle-finding constructor 
does G have a directed cycle ? 
vertices on a cycle (if one exists) 



marked[] 
012345 ... 

dfs(0) 

dfs(5) 1 0 

dfs(4) 100001 

dfs(3) 100011 

check 5 100111 


edgeTo[] 

0 1 2 3 4 5 

0 

5 0 
4 5 

-4 5 0 


Finding a directed cycle in a digraph 


onStack[] 

0 1 2 3 4 5 

1 0 0 0 0 C 
1 0 0 0 0 1 
1 0 0 0 1 

10 0 1 l(T) 
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Finding a directed cycle 


public class DirectedCycle 

{ 

private boolean[] marked; 
private int[] edgeTo; 

private Stack<Integer> cycle; // vertices on a cycle (if one exists) 

private boolean[] onStack; // vertices on recursive call stack 


public DirectedCycle(Digraph G) 

{ 

onStack = new boolean[G.V()]; 
edgeTo = new int[G.V()]; 
marked = new boolean[G.V()]; 
for (int v = 0; v < G.V(); v++) 
if (!marked[v]) dfs(G, v); 

} 

private void dfs(Digraph G, int v) 

{ 

onStack[v] = true; 
marked[v] = true; 
for (int w : G.adj(v)) 

if (this.hasCycle()) return; 
else if (!marked[w]) 

{ edgeTo[w] = v; dfs(G, w); } 

else if (onStack[w]) 

{ 

cycle = new Stack<Integer>(); 
for (int x = v; x != w; x = edgeTo[x]) 
cycle.push(x); 
cycle.push(w); 
cycle.push(v); 

} 

onStack[v] = false; 



3 3 

4 4 3 

5 4 5 4 3 

3 5 4 3 5 4 3 



Trace of cycle computation 


public boolean hasCycleO 
{ return cycle != null; } 

public Iterable<Integer> cycle() 
{ return cycle; } 


This class adds to our standard recursive df s () a boolean array onStack [] to keep track of the verti¬ 
ces for which the recursive call has not completed. When it finds an edge v->w to a vertex w that is on 
the stack, it has discovered a directed cycle, which it can recover by following edgeTo [] links. 
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When executing df s (G, v) , we have followed a directed path from the source to v. To 
keep track of this path, Di rectedCycle maintains a vertex-indexed array onStack[] 
that marks the vertices on the recursive call stack (by setting onStack[v] to true 
on entry to dfs(G, v) and to false on exit). Di rectedCycle also maintains an 
edgeTo[] array so that it can return the cycle when it is detected, in the same way as 
DepthFi rstPaths (page 536) and BreadthFi rstPaths (page 540) return paths. 

Depth-first orders and topological sort. Precedence-constrained scheduling amounts 
to computing a topological order for the vertices of a DAG, as in this API: 

public class Topological 

Topol ogi cal (Digraph G) topological-sorting constructor 
boolean isDAGO is G a DAG? 

Iterable<Integer> orderO vertices in topological order 

API for topological sorting 


Proposition E. A digraph has a topological order if and only if it is a DAG. 

Proof: If the digraph has a directed cycle, it has no topological order. Conversely, 
the algorithm that we are about to examine computes a topological order for any 
given DAG. 


Remarkably, it turns out that we have already seen an algorithm for topological sort: a 
one-line addition to our standard recursive DFS does the job! To convince you of this 
fact, we begin with the class DepthFi rstOrder on page 580. It is based on the idea that 
depth-first search visits each vertex exactly once. If we save the vertex given as argument 
to the recursive dfs() in a data structure, then iterate through that data structure, we 
see all the graph vertices, in order determined by the nature of the data structure and 
by whether we do the save before or after the recursive calls. Three vertex orderings are 
of interest in typical applications: 

■ Preorder : Put the vertex on a queue before the recursive calls. 

■ Postorder : Put the vertex on a queue after the recursive calls. 

■ Reverse postorder : Put the vertex on a stack after the recursive calls. 

A trace of Depth Fi rstOrde r for our sample DAG is given on the facing page. It is simple 
to implement and supports pre(), post(), and reversePostO methods that are use¬ 
ful for advanced graph-processing algorithms. For example, orderO in Topological 
consists of a call on reversePostQ. 
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preorder 
ts order of 
dfsQ calls 


\ 


pre 


postorder 
is order 
in which 
vertices 
are done 


I 


post 


reversePost 


dfs(O) 

dfs(5) 

dfs(4) 

4 done 

5 done 
dfs(l) 

1 done 
dfs(6) 

dfs(9) 
dfs(ll) 
dfs(12) 
12 done 
11 done 
dfs(10) 

10 done 
check 12 
9 done 
check 4 

6 done 
0 done 
check 1 
dfs(2) 

check 0 
dfs(3) 
check 5 
3 done 
2 done 
check 3 
check 4 
check 5 
check 6 
dfs(7) 
check 6 

7 done 
dfs(8) 

check 7 

8 done 
check 9 
check 10 
check 11 
check 12 


0 

0 5 
0 5 4 


0 5 4 1 


queue 

/ 


0 5 4 1 6 
0 5 4 1 6 9 
0 5 4 1 6 9 11 
0 5 4 1 6 9 11 12 


0 5 4 1 6 9 11 12 10 


0 5 4 1 6 9 11 12 10 2 
0 5 4 1 6 9 11 12 10 2 3 


054169 11 12 10 237 

054169 11 12 10 2378 


4 

4 5 


queue 



4 5 1 


1 5 4 


stack 

/ 


4 5 1 12 

4 5 1 12 11 

4 5 1 12 11 10 

4 5 1 12 11 10 9 

4 5 1 12 11 10 9 6 

4 5 1 12 11 10 9 6 0 


12 1 5 4 
11 12 1 5 4 

10 11 12 1 5 4 

9 10 11 12 1 5 4 

6 9 10 11 12 1 5 4 
0 6 9 10 11 12 1 5 4 


4 5 1 12 11 10 9 6 0 3 3 0 6 9 10 11 12 1 5 4 

45112111096032 23069 10 11 12 154 


45112 11 10 960327 723069 10 11 12 154 


45112 11 10 9603278 8723069 10 11 12 15' 

t 

reverse 

postorder 


Computing depth-first orders in a digraph (preorder, postorder, and reverse postorder) 
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Depth-first search vertex ordering in a digraph 


public class DepthFirstOrder 

{ 

private booleanf] marked; 

private Queue<Integer> pre; // vertices in preorder 

private Queue<Integer> post; // vertices in postorder 

private Stack<Integer> reversePost; // vertices in reverse postorder 

public DepthFirstOrder(Digraph G) 

{ 

pre = new Queue<Integer>(); 

post = new Queue<Integer>(); 

reversePost = new Stack<Integer>(); 

marked = new boolean[G.V()]; 

for (int v = 0; v < G.V(); v++) 
if (!marked[v]) dfs(G, v); 

} 

private void dfs(Digraph G, int v) 

{ 

pre.enqueue(v); 

marked[v] = true; 
for (int w : G.adj(v)) 
if (!marked[w]) 
dfs(G, w); 

post.enqueue(v); 
reversePost.push(v); 

} 

public Iterable<Integer> pre() 

{ return pre; } 

public Iterable<Integer> post() 

{ return post; } 

public Iterable<Integer> reversePostO 
{ return reversePost; } 

} 


This class enables clients to iterate through the vertices in various orders defined by depth-first search. 
This ability is very useful in the development of advanced digraph-processing algorithms, because the 
recursive nature of the search enables us to prove properties of the computation (see, for example, 
proposition f). 
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ALGORITHM 4.5 Topological sort 


public class Topological 

{ 

private Iterable<Integer> order; // topological order 

public Topological(Digraph G) 

{ 

DirectedCycle cyclefinder = new DirectedCycle(G); 
if (!cyclefinder.hasCycle()) 

{ 

DepthFirstOrder dfs = new DepthFirstOrder(G); 
order = dfs. reversePostO ; 

} 

} 

public Iterable<Integer> order() 

{ return order; } 

public boolean isDAGO 
{ return order == null; } 

public static void main(String[] args) 

{ 

String filename = args[0]; 

String separator = args[l]; 

Symbol Digraph sg = new SymbolDigraph(filename, separator); 
Topological top = new Topological (sg.GO) ; 

for (int v : top.orderO) 

StdOut.println(sg.name(v)); 

} 

} 


This DepthFi rstOrder and Di rectedCycl e client returns a topological order for a DAG. The test 
client solves the precedence-constrained scheduling problem for a Symbol Digraph. The instance 
method order() returns nul 1 if the given digraph is not a DAG and an iterator giving the vertices in 
topological order otherwise. The code for Symbol Di graph is omitted because it is precisely the same 
as for Symbol Graph (page 552), with Digraph replacing Graph everywhere. 
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Proposition F. Reverse postorder in a DAG is a topological sort. 

Proof: Consider any edge v->w. One of the following three cases must hold when 
dfs(v) is called (see the diagram on page 583): 

■ df s (w) has already been called and has returned (w is marked). 

■ df s (w) has not yet been called (w is unmarked), so v->w will cause df s (w) to 
be called (and return), either directly or indirectly, before dfs(v) returns. 

■ df s (w) has been called and has not yet returned when df s (v) is called. The 
key to the proof is that this case is impossible in a DAG, because the recursive 
call chain implies a path from w to v and v->w would complete a directed 
cycle. 

In the two possible cases, dfs(w) is done before dfs(v), so w appears before v in 
postorder and after v in reverse postorder. Thus, each edge v->w points from a ver¬ 
tex earlier in the order to a vertex later in the order, as desired. 


% more jobs.txt 

Algorithms/Theoretical CS/Databases/Scientific Computing 
Introduction to CS/Advanced Programming/Algorithms 
Advanced Programming/Scientific Computing 
Scientific Computing/Computational Biology 

Theoretical CS/Computational Biology/Artificial Intelligence 
Linear Algebra/Theoretical CS 
Calculus/Linear Algebra 

Artificial Intelligence/Neural Networks/Robotics/Machine Learning 
Machine Learning/Neural Networks 

% java Topological jobs.txt "/" 

Calculus 
Linear Algebra 
Introduction to CS 
Advanced Programming 
Algorithms 
Theoretical CS 
Artificial Intelligence 
Robotics 

Machine Learning 
Neural Networks 
Databases 

Scientific Computing 
Computational Biology 
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Topological (algorithm 4.5 on page 581) is 
an implementation that uses depth-first search to 
topologically sort a DAG. A trace is given at right. 

Proposition G. With DFS, we can topologically 
sort a DAG in time proportional to V+E. 

Proof: Immediate from the code. It uses one 
depth-first search to ensure that the graph has 
no directed cycles, and another to do the re¬ 
verse postorder ordering. Both involve examin¬ 
ing all the edges and all the vertices, and thus 
take time proportional to V+E. 


Despite the simplicity of this algorithm, it escaped 
attention for many years, in favor of a more intui¬ 
tive algorithm based on maintaining a queue of 
sources (see exercise 4 . 2 . 30 ). 

in practice, topological sorting and cycle detec¬ 
tion go hand in hand, with cycle detection play¬ 
ing the role of a debugging tool. For example, in 
a job-scheduling application, a directed cycle in 
the underlying digraph represents a mistake that 
must be corrected, no matter how the schedule was 
formulated. Thus, a job-scheduling application is 
typically a three-step process: 

■ Specify the tasks and precedence constraints. 

■ Make sure that a feasible solution exists, by 
detecting and removing cycles in the under¬ 
lying digraph until none exist. 

■ Solve the scheduling problem, using topo¬ 
logical sort. 

Similarly, any changes in the schedule can be 
checked for cycles (using Di rectedCycl e), then a 
new schedule computed (using Topol ogi cal). 




df s (5) for 0’s unmarked 
neighbor 5 is done 
before dfs(O) is done 
so 0->5 points up 


12 done 
11 done 
dfs(10) 

10 done 
check 12 
done 



dfs(6) for 7 s marked 
neighbor 6 was done 
before dfs(7) is done 
so 7 ->6 points up 


all edges point up; 
turn upside down 
for a topological sort 


reverse postorder 
is reverse of order 
in which vertices 
are done (read up) 


Reverse postorder in a DAG is a topological sort 
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Strong connectivity in digraphs We have been careful to maintain a distinction 
between reachability in digraphs and connectivity in undirected graphs. In an undirect¬ 
ed graph, two vertices v and w are connected if there is a path connecting 
them—we can use that path to get from v to w or to get from w to v. In 
a digraph, by contrast, a vertex w is reachable from a vertex v if there is 
a directed path from v to w, but there may or may not be a directed path 
back to v from w. To complete our study of digraphs, we consider the 
natural analog of connectivity in undirected graphs. 


Several examples of strongly connected graphs are given in the figure 
at left. As you can see from the examples, cycles play an important role 
in understanding strong connectivity. Indeed, recalling that a general 
directed cycle is a directed cycle that may have repeated vertices, it is easy 
to see that two vertices are strongly connected if and only if there exists a 
general directed cycle that contains them both. (Proof: compose the paths 
from v to w and from w to v.) 

Strong components. Like connectivity in undirected graphs, strong connectivity in di¬ 
graphs is an equivalence relation on the set of vertices, as it has the following properties: 

■ Reflexive: Every vertex v is strongly connected to itself. 

■ Symmetric: If v is strongly connected to w, then w is strongly connected to v. 

■ Transitive: If v is strongly connected to w and w is strongly connected to x, then v 
is also strongly connected to x. 

As an equivalence relation, strong connectivity partitions the vertices into equivalence 
clasees. The equivalence classes are maximal subsets 
of vertices that are strongly connected to one anoth¬ 
er, with each vertex in exactly one subset. We refer 
to these subsets as strongly connected components, 
or strong components for short. Our sample digraph 
ti nyDG. txt has five strong components, as shown 
in the diagram at right. A digraph with V vertices 
has between 1 and V strong components—a strongly 



A digraph and its strong components 


Definition. Two vertices v and w are strongly connected if they are 
mutually reachable: that is, if there is a directed path from v to w 
and a directed path from w to v. A digraph is strongly connected if all 
its vertices are strongly connected to one another. 


o 

I 

°5 



Strongly connected digraphs 
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connected digraph has 1 strong component and a DAG has V strong components. 
Note that the strong components are defined in terms of the vertices, not the edges. 
Some edges connect two vertices in the same strong component; some other edges con¬ 
nect vertices in different strong components. The latter are not found on any directed 
cycle. Just as identifying connected components is typically important in processing 
undirected graphs, identifying strong components is typically important in processing 
digraphs. 

Examples of applications. Strong connectivity is a useful abstraction in understand¬ 
ing the structure of a digraph, highlighting inter¬ 
related sets of vertices (strong components). For 
example, strong components can help textbook 
authors decide which topics should be grouped to¬ 
gether and software developers decide how to orga¬ 
nize program modules. The figure below shows an 
example from ecology. It illustrates a digraph that 
models the food web connecting living organisms, 
where vertices represent species and an edge from 
one vertex to another indicates that an organism of 
the species indicated by the point from vertex con¬ 
sumes organisms of the species indicated by the 
point to vertex for food. Scientific studies on such digraphs (with carefully chosen sets 
of species and carefully documented relationships) play an important role in helping 
ecologists answer basic questions about ecological systems. Strong components in such 

digraphs can help ecologists understand en¬ 
ergy flow in the food web. The figure on page 
591 shows a digraph model of web content, 
where vertices represent pages and edges rep¬ 
resent hyperlinks from one page to another. 
Strong components in such a digraph can help 
network engineers partition the huge number 
of pages on the web into more manageable siz¬ 
es for processing. Further properties of these 
applications and other examples are addressed 
in the exercises and on the booksite. 



application vertex edge 


web 

page 

hyperlink 

textbook 

topic 

reference 

software 

module 

call 

food web 

organism 

predator-prey 

relationship 


Typical strong-component applications 
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Accordingly, we need the following API, the analog for digraphs of CC (page 543): 
pub!ic cl ass SCC 


SCC(Digraph G) 


preprocessing constructor 


boolean stronglyConnected(int v, int w) are v and w strongly connected? 


int countQ 


number of strong components 

component identifier for v 
(between 0 and count()-l) 


int id (int v) 


API for strong components 


A quadratic algorithm to compute strong components is not difficult to develop (see 
exercise 4 . 2 . 23 ), but (as usual) quadratic time and space requirements are prohibitive 
for huge digraphs that arise in practical applications like the ones just described. 

Kosaraju’s algorithm. We saw in CC (algorithm 4.3 on page 544) that computing con¬ 
nected components in undirected graphs is a simple application of depth-first search. 
How can we efficiently compute strong components in digraphs? Remarkably, the im¬ 
plementation Kosaraj uSCC on the facing page does the job with just a few lines of code 
added to CC, as follows: 

■ Given a digraph G, use DepthFi rstOrder to compute the reverse postorder of 
its reverse, G R . 

■ Run standard DFS on G, but consider the unmarked vertices in the order just 
computed instead of the standard numerical order. 

■ All vertices reached on a call to the recursive df s () from the constructor are in a 
strong component (!), so identify them as in CC. 

DFS in G (KosarajuSCC) DFS in G R (DepthFi rstOrder) 


' df s (s) "is reachable from s ■ dfs(v) 


premise is that 


so G must have 
a path 

dfs(v) from s to v 



dfs(s) 



G r must have 
9 a path 
from s to v 


v d- 


v must be done 


dfs(v) 


s done 


v done 



s done 


s done 


v done 


t 


impossible since G R 
has a path from v to s 


Proof of correctness for Kosaraju's algorithm 
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ALGORITHM 4.6 Kosaraju's algorithm for computing strong components 


public class KosarajuSCC 

{ 

private booleanf] marked; // reached vertices 
private int[] id; // component identifiers 

private int count; // number of strong components 

public KosarajuSCC(Digraph G) 

{ 

marked = new boolean[G.V()]; 
id = new int[G.V()]; 

DepthFi rstOrder order = new DepthFi rstOrder(G. reverseO) ; 
for (int s : order. reversePostO) 

if (!marked[s]) 

{ dfs(G, s); count++; } 

} 

% java KosarajuSCC tmyDG.txt 


private void dfs(Digraph G, int v) 5 components 

{ 1 

markedfv] = true; 05432 

id[v] = count; 11 12 9 10 

for (int w : G.adj(v)) 6 

if (!marked[w]) 8 7 

dfs(G, w); 

} 


public boolean stronglyConnected(int v, int w) 
{ return id[v] == id[w]; } 

public int id(int v) 

{ return id[v]; } 

public int countO 
{ return count; } 


} 


This implementation differs from CC (algorithm 4 . 3 ) only in the highlighted code (and in the im¬ 
plementation of mai n() where we use the code on page 543, with Graph changed to Digraph,, and CC 
changed to KosarajuSCC). To find strong components, it does a depth-first search in the reverse 
digraph to produce a vertex order (reverse postorder of that search) for use in a depth-first search of 
the given digraph. 
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Kosaraju’s algorithm is an extreme example of a method that is easy to code but dif¬ 
ficult to understand. Despite its mysterious nature, if you follow the proof of the fol¬ 
lowing proposition step by step, with reference to the diagram on page 586, you can be 
convinced that the algorithm is correct: 

Proposition H. In a DFS of a digraph G where marked vertices are considered in 
reverse postorder given by a DFS of the digraph’s reverse G R (Kosaraju’s algorithm), 
the vertices reached in each call of the recursive method from the constructor are 
in a strong component. 

Proof: First, we prove by contradiction that every vertex v that is strongly connected 
to s is reached by the call dfs(G, s) in the constructor. Suppose a vertex v that is 
strongly connected to s is not reached by dfs(G, s). Since there is a path from s to 
v, v must have been previously marked. But then, since there is a path from v to s, s 
would have been marked during the call df s (G, v) and the constructor would not 
calldfsCG, s), a contradiction. 

Second, we prove that every vertex v reached by the call df s (G, s) in the construc¬ 
tor is strongly connected to s. Let v be a vertex reached by the call df s (G, s). Then, 
there is a path from s to v in G, so it remains only to prove that there is a path from 
v to s in G. This statement is equivalent to saying that there is a path from s to v in 
G R , so it remains only to prove that there is a path from s to v in G R . 

The crux of the proof is that the reverse postorder construction implies that 
df s (G, v) must have been done before df s (G, s) during the DFS of G R , leaving 
just two cases to consider for df s (G, v): it might have been called 

■ before df s (G, s) was called (and also done before df s (G, s) was called) 

■ after dfs(G, s) was called (and done before dfs(G, s) was done) 

The first of these is not possible because there is a path from v to s in G R ; the second 
implies that there is a path from s to v in G R , completing the proof. 


A trace of Kosaraju’s algorithm for tinyDG.txt is shown on the facing page. To the 
right of each DFS trace appears a drawing of the digraph, with vertices appearing in 
the order they are done. Thus reading up the reverse digraph drawing on the left gives 
reverse postorder, the order in which unmarked vertices are checked in the DFS of the 
original digraph. As you can see from the diagram, the second DFS calls df s (1) (which 
marks vertex 1) then calls df s (0) (which marks 5, 4, 3, and 2), then checks 2,4, 5, and 
3, then calls dfs(ll) (which marks 11,12, 9, and 10), then checks 9,12, and 10, then 
calls dfs (6) (which marks 6), and finally dfs (7), which marks 7 and 8. 
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DFS in reverse digraph (ReversePost) 



check unmarked vertices in the order 


DFS in original digraph 



check unmarked vertices in the order 


0123456789 10 11 12 


102453 11 9 12 10 678 


dfs(0) 
dfs(6) 
dfs(7) 
dfs(8) 
check 7 
8 done 
7 done 
6 done 
dfs(2) 
dfs(4) 
dfs(11) 
dfs(9) 
dfs(12) 
check 11 
dfs(10) 
check 9 
10 done 
12 done 
check 8 
check 6 
9 done 
11 done 
check 6 
dfs(5) 
dfs(3) 
check 4 
check 2 
3 done 
check 0 
5 done 
4 done 
check 3 
2 done 
0 done 
dfs(l) 
check 0 
1 done 
check 2 
check 3 
check 4 
check 5 
check 6 
check 7 
check 8 
check 9 
check 10 
check 11 
check 12 



t 


reverse 
postorder 
for use 
in second 
dfs() 

(read up) 



dfs(5) 

dfs(4) 


dfs(3) 
check 5 
dfs(2) 
check 0 
check 3 
2 done 
3 done 
check 2 
4 done 
5 done 
check 1 

^0 done _/ 

check 2 
check 4 
check 5 

check 3 _ 

f dfs(11) \ 

check 4 
dfs(12) 
dfs(9) 
check 11 
dfs(10) 
check 12 
10 done 
9 done 
12 done 

y 11 done _ 

check 9 
check 12 
check 10 

( dfs(6) A 
check 9 
check 4 
check 0 

V6 done _ J 

f dfs(7) \ 

check 6 
dfs(8) 
check 7 
check 9 
8 done 

y7 done _ 

check 8 



Kosaraju's algorithm for finding strong components in digraphs 
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A larger example, a very small subset of a digraph model of the web, is shown on the 
facing page. 

kosaraju’s algorithm solves the following analog of the connectivity problem for 
undirected graphs that we first posed in chapter i and reintroduced in section 4.1 
(page 534): 

Strong connectivity. Given a digraph, support queries of the form: Are two given 
vertices strongly connected ? and How many strong components does the digraph 
have ? 

That we can solve this problem in digraphs as efficiently as the corresponding con¬ 
nectivity problem in undirected graphs was an open research problem for some time 
(resolved by R. E. Tarjan in the late 1970s). That such a simple solution is now available 
is quite surprising. 

Proposition I. Kosaraju’s algorithm uses preprocessing time and space proportion¬ 
al to V+E to support constant-time strong connectivity queries in a digraph. 

Proof: The algorithm computes the reverse of the digraph and does two depth-first 
searches. Each of these three steps takes time proportional to V+E. The reverse copy 
of the digraph uses space proportional to V+E. 


Reachability revisited. With CC for undirected graphs, we can infer from the fact that 
two vertices v and w are connected that there is a path from v to w and a path (the same 
one) from w to v. With Kosara j uCC, we can infer from the fact that v and w are strongly 
connected that there is a path from v to w and a path (a different one) from w to v. But 
what about pairs of vertices that are not strongly connected? There may be a path from 
v to w or a path from w to v or neither, but not both. 

All-pairs reachability. Given a digraph, support queries of the form Is there a 

directed path from a given vertex v to another given vertex w? 

For undirected graphs, the corresponding problem is equivalent to the connectivity 
problem; for digraphs, it is quite different from the strong connectivity problem. Our 
CC implementation uses linear preprocessing time to support constant-time answers 
to such queries for undirected graphs. Can we achieve this performance for digraphs? 
This seemingly innocuous question has confounded experts for decades. To better 


www.it-ebooks.info 



4.2 ■ 


Directed Graphs 


591 



How many strong components are there in this digraph? 
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understand the challenge, consider the diagram at 
left, which illustrates the following fundamental 
concept: 


By convention, every vertex is reachable from itself, 
so the transitive closure has V self-loops. Our sam¬ 
ple digraph has just 13 directed edges, but its transi¬ 
tive closure has 102 out of a possible 169 directed 
edges. Generally, the transitive closure of a digraph 
has many more edges than the digraph itself, and 
it is not at all unusual for a sparse graph to have a 
dense transitive closure. For example, the transitive 
closure of a V-vertex directed cycle, which has V di¬ 
rected edges, is a complete digraph with V 2 directed 
edges. Since transitive closures are typically dense, 
we normally represent them with a matrix of boolean values, where the entry in row v 
and column w is true if and only if w is reachable from v. Instead of explicitly comput¬ 
ing the transitive closure, we use depth-first search to implement the following API: 

public class TransitiveClosure 

TransitiveClosure(Digraph G) preprocessing constructor 
boolean reachabl e(i nt v, int w) is w reachable from v? 

API for all-pairs reachability 

The code at the top of the next page is a straightforward implementation that uses 
Di rectedDFS (algorithm 4 . 4 ). This solution is ideal for small or dense digraphs, but 
it is not a solution for the large digraphs we might encounter in practice because the 
constructor uses space proportional to V 2 and time proportional to V (V+E): each of the 
V Di rectedDFS objects takes space proportional to V (they all have marked [] arrays of 
size V and examine E edges to compute the marks). Essentially, TransitiveClosure 


Definition. The transitive closure of a digraph 
G is another digraph with the same set of ver¬ 
tices, but with an edge from v to w in the tran¬ 
sitive closure if and only if w is reachable from 
v in G. 



2 3 4 5 6 7 8 9 10 11 12 

T T T T 


T 
T 
T 
T 
T 
T 

T T T 
T T T 


original edge 
^ {red) 

^ self-loop 
* (gray) 


T 

T T 
T T T 
T 
T 


T T 
T T 
T T 


T T 
T T 


12 is 

reachable 
from 6 


Transitive closure 
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computes and stores the transitive closure of G, to support constant-time queries— 
row v in the transitive closure matrix is the marked [] array for the vth entry in the 
Di rectedDFS[] in TransitiveClosure. Can we support constant-time queries with 
substantially less preprocessing time and substantially less space? A general solution 
that achieves constant-time queries with 
substantially less than quadratic space is 
an unsolved research problem, with im¬ 
portant practical implications: for exam¬ 
ple, until it is solved, we cannot hope to 
have a practical solution to the all-pairs 
reachability problem for a giant digraph 
such as the web graph. 


public class TransitiveClosure 
{ 

private DirectedDFS[] all; 
TransitiveClosure(Digraph G) 

{ 

all = new DirectedDFS[G.V()]; 
for (int v = 0; v < G.V(); v++) 
all[v] = new DirectedDFSfG, v); 

} 

boolean reachable(int v, int w) 

{ return all[v].marked(w); } 


All-pairs reachability 
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Summary In this section, we have introduced directed edges and digraphs, empha¬ 
sizing the relationship between digraph processing and corresponding problems for 
undirected graphs, as summarized in the following list of topics: 

■ Digraph nomenclature 

■ The idea that the representation and approach are essentially the same as for 
undirected graphs, but some digraph problems are more complicated 

■ Cycles, DAGs, topological sort, and precedence-constrainted scheduling 

■ Reachability, paths, and strong connectivity in digraphs 

The table below summarizes the implementations of digraph algorithms that we have 
considered (all but one of the algorithms are based on depth-first search). The prob¬ 
lems addressed are all simply stated, but the solutions that we have considered range 
from easy adaptations of corresponding algorithms for undirected graphs to an inge¬ 
nious and surprising solution. These algorithms are a starting point for several of the 
more complicated algorithms that we consider in section 4 . 4 , when we consider edge- 
weighted digraphs. 


problem 

solution 

reference 

single- and multiple-source reachability 

DirectedDFS 

page 571 

single-source directed paths 

DepthFirstDirectedPaths 

page 573 

single-source shortest directed paths 

BreadthFirstDirectedPaths 

page 573 

directed cycle detection 

DirectedCycle 

page 577 

depth-first vertex orders 

DepthFirstOrder 

page 580 

precedence-constrained scheduling 

Topological 

page 581 

topological sort 

Topol ogical 

page 581 

strong connectivity 

KosarajuSCC 

page 587 

all-pairs reachability 

TransitiveClosure 

page 593 


Digraph-processing problems addressed in this section 
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Q&A 


Q. Is a self-loop a cycle? 

A. Yes, but no self-loop is needed for a vertex to be reachable from itself. 
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| EXERCISES 


4 . 2.1 What is the maximum number of edges in a digraph with V vertices and no paral¬ 
lel edges? What is the minimum number of edges in a digraph with V vertices, none of 
which are isolated? 

4 . 2.2 Draw, in the style ofthe figure in the text (page 524), the adja¬ 
cency lists built by Di g raph’s input stream constructor for the file 
ti nyDGex2 . txt depicted at left. 

4 . 2.3 Create a copy constructor for Digraph that takes as input 
a digraph G and creates and initializes a new copy of the digraph. 
Any changes a client makes to G should not affect the newly created 
digraph. 

4 . 2.4 Add a method has Edge O to Digraph which takes two i nt 
arguments v and w and returns true if the graph has an edge v->w, 
fal se otherwise. 

4 . 2.5 Modify Di graph to disallow parallel edges and self-loops. 

4 . 2.6 Develop a test client for Di g raph. 

4 . 2.7 The indegree of a vertex in a digraph is the number of directed edges that point to 
that vertex. The outdegree of a vertex in a digraph is the number of directed edges that 
emanate from that vertex. No vertex is reachable from a vertex of outdegree 0, which is 
called a sink ; a vertex of indegree 0, which is called a source, is not reachable from any 
other vertex. A digraph where self-loops are allowed and every vertex has outdegree 1 
is called a map (a function from the set of integers from 0 to V-l onto itself). Write a 
program Degrees . java that implements the following API: 


tinyDGex2.txt 



public class Degrees 


int 

int 

Iterable<Integer> 

Iterable<Integer> 

boolean 


Degrees(Digraph G) 
indegree(int v) 
outdegreefint v) 
sourcesO 
sinks() 
i sMapQ 


constructor 
indegree ofv 
outdegree ofv 
sources 
sinks 

is G a map? 
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4.2.8 Draw all the nonisomorphic DAGs with two, three, four, and five vertices (see 
exercise 4 . 1 . 28 ). 

4.2.9 Write a method that checks whether or not a given permutation of a DAG’s ver¬ 
tices is a topological order of that DAG. 

4.2.10 Given a DAG, does there exist a topological order that cannot result from ap¬ 
plying a DFS-based algorithm, no matter in what order the vertices adjacent to each 
vertex are chosen? Prove your answer. 

4.2.1 1 Describe a family of sparse digraphs whose number of directed cycles grows 
exponentially in the number of vertices. 

4.2.12 How many edges are there in the transitive closure of a digraph that is a simple 
directed path with V vertices and V-l edges? 

4.2.13 Give the transitive closure of the digraph with ten vertices and these edges: 
3->7 l->4 7->8 0->5 5->2 3->8 2->9 0->6 4->9 2->6 6->4 

4.2.14 Prove that the strong components in G R are the same as in G. 

4.2.15 What are the strong components of a DAG? 

4.2.16 What happens if you run Kosaraju’s algorithm on a DAG? 

4.2.17 True or false: The reverse postorder of a graph’s reverse is the same as the post¬ 
order of the graph. 

4.2.18 Compute the memory usage of a Digraph with V vertices and E edges, under 
the memory cost model of section 1 . 4 . 
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| CREATIVE PROBLEMS 

4.2.19 Topological sort and BFS. Explain why the following algorithm does not neces¬ 
sarily produce a topological order: Run BFS, and label the vertices by increasing dis¬ 
tance to their respective source. 

4.2.20 Directed Eulerian cycle. An Eulerian cycle is a directed cycle that contains each 
edge exactly once. Write a graph client Euler that finds an Eulerian cycle or reports that 
no such tour exists. Hint: Prove that a digraph G has a directed Eulerian cycle if and 
only if G is connected and each vertex has its indegree equal to its outdegree. 

4.2.21 LCA of a DAG. Given a DAG and two vertices v and w, find the lowest common 
ancestor (LCA) of v and w. The LCA of v and w is an ancestor of v and w that has no 
descendants that are also ancestors of v and w. Computing the LCA is useful in multiple 
inheritance in programming languages, analysis of genealogical data (find degree of 
inbreeding in a pedigree graph), and other applications. Hint: Define the height of a 
vertex v in a DAG to be the length of the longest path from a root to v. Among vertices 
that are ancestors of both v and w, the one with the greatest height is an LCA of v and w. 

4.2.22 Shortest ancestral path. Given a DAG and two vertices v and w, find the shortest 
ancestral path between v and w. An ancestral path between v and w is a common ancestor 
x along with a shortest path from v to x and a shortest path from w to x. The shortest 
ancestral path is the ancestral path whose total length is minimized. Warmup: Find a 
DAG where the shortest ancestral path goes to a common ancestor x that is not an LCA. 
Hint: Run BFS twice, once from v and once from w. 

4.2.23 Strong component. Describe a linear-time algorithm for computing the strong 
connected component containing a given vertex v. On the basis of that algorithm, de¬ 
scribe a simple quadratic algorithm for computing the strong components of a digraph. 

4.2.24 Hamiltonian path in DAGs. Given a DAG, design a linear-time algorithm to 
determine whether there is a directed path that visits each vertex exactly once. 

Answer: Compute a topological sort and check if there is an edge between each con¬ 
secutive pair of vertices in the topological order. 

4.2.25 Unique topological ordering. Design an algorithm to determine whether a di¬ 
graph has a unique topological ordering. Hint: A digraph has a unique topological 
ordering if and only if there is a directed edge between each pair of consecutive vertices 
in the topological order (i.e., the digraph has a Hamiltonian path). If the digraph has 
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multiple topological orderings, then a second topological order can be obtained by 
swapping a pair of consecutive vertices. 

4.2.26 2-satisfiability. Given boolean formula in conjunctive normal form with M 
clauses and N literals such that each clause has exactly two literals, find a satisfying as¬ 
signment (if one exists). Hint: Form the implication digraph with 2 N vertices (one per 
literal and its negation). For each clause x + y, include edges from y' to x and from x' 
to y. To satisfy the clause x+y, (i) if y is false, then x is true and (ii) if x is false, then y is 
true. Claim: The formula is satisfiable if and only if no variable x is in the same strong 
component as its negation x'. Moreover, a topological sort of the kernel DAG (contract 
each strong component to a single vertex) yields a satisfying assignment. 

4.2.27 Digraph enumeration. Show that the number of different V-vertex digraphs 
with no parallel edges is 2 V ~. (How many digraphs are there that contain V vertices and 
E edges?) Then compute an upper bound on the percentage of 20-vertex digraphs that 
could ever be examined by any computer, under the assumptions that every electron 
in the universe examines a digraph every nanosecond, that the universe has fewer than 
10 80 electrons, and that the age of the universe will be less than 10 20 years. 

4.2.28 DAG enumeration. Give a formula for the number of V-vert ex DAGs with E 
edges. 

4.2.29 Arithmetic expressions. Write a class that evaluates DAGs that represent arith¬ 
metic expressions. Use a vertex-indexed array to hold values corresponding to each 
vertex. Assume that values corresponding to leaves have been established. Describe a 
family of arithmetic expressions with the property that the size of the expression tree 
is exponentially larger than the size of the corresponding DAG (so the running time of 
your program for the DAG is proportional to the logarithm of the running time for the 
tree). 

4.2.30 Queue-based topological sort. Develop a topological sort implementation that 
maintains a vertex-indexed array that keeps track of the indegree of each vertex. Initial¬ 
ize the array and a queue of sources in a single pass through all the edges, as in exercise 
4 . 2 . 7 . Then, perform the following operations until the source queue is empty: 

■ Remove a source from the queue and label it. 

■ Decrement the entries in the indegree array corresponding to the destination 
vertex of each of the removed vertex’s edges. 
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CREATIVE PROBLEMS (continued) 


■ If decrementing any entry causes it to become 0, insert the corresponding vertex 
onto the source queue. 

4.2.31 Euclidean digraphs. Modify your solution to exercise 4 . 1.37 to create an API 
Eucl i deanDigraph for graphs whose vertices are points in the plane, so that you can 
work with graphical representations. 
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EXPERIMENTS 

4.2.32 Random digraphs. Write a program ErdosRenyiDigraph that takes integer 
values V and E from the command line and builds a digraph by generating E random 
pairs of integers between 0 andV— 1. Note: This generator produces self-loops and par¬ 
allel edges. 

4.2.33 Random simple digraphs. Write a program RandomDigraph that takes integer 
values V and E from the command line and produces, with equal likelihood, each of the 
possible simple digraphs with V vertices and E edges. 

4.2.34 Random sparse digraphs. Modify your solution to exercise 4 . 1.41 to create 
a program RandomSparseDigraph that generates random sparse digraphs for a well- 
chosen set of values of V and E that you can use it to run meaningful empirical tests. 

4.2.35 Random Euclidean digraphs. Modify your solution to exercise 4 . 1.42 to create 
a Eucl ideanDi graph client RandomEucl i deanDi graph that assigns a random direc¬ 
tion to each edge. 

4.2.36 Random grid digraphs. Modify your solution to exercise 4 . 1.43 to create a 
Eucl i deanDi Graph client RandomGri dDi graph that assigns a random direction to each 
edge. 

4.2.37 Real-world digraphs. Find a large digraph somewhere online—perhaps a 
transaction graph in some online system, or a digraph defined by links on web pages. 
Write a program RandomRealDigraph that builds a graph by choosing V vertices at 
random and E directed edges at random from the subgraph induced by those vertices. 

4.2.38 Real-world DAG. Find a large DAG somewhere online—perhaps one defined 
by class-definition dependencies in a large software system, or by directory links in a 
large file system. Write a program RandomReal DAG that builds a graph by choosing V 
vertices at random and E directed edges at random from the subgraph induced by those 
vertices. 
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EXPERIMENTS (continued) 


Testing all algorithms and studying all parameters against all graph models is unrealistic. 
For each problem listed below, write a client that addresses the problem for any given input 
graph, then choose among the generators above to run experiments for that graph model. 
Use your judgment in selecting experiments, perhaps in response to results of previous 
experiments. Write a narrative explaining your results and any conclusions that might be 
drawn. 


4.2.39 Reachability. Run experiments to determine empirically the average number of 
vertices that are reachable from a randomly chosen vertex, for various digraph models. 

4.2.40 Path lengths in DFS. Run experiments to determine empirically the probability 
that DepthFi rstDi rectedPaths finds a path between two randomly chosen vertices 
and to calculate the average length of the paths found, for various random digraph 
models. 


4.2.41 Path lengths in BFS. Run experiments to determine empirically the probability 
that BreadthFi rstDi rectedPaths finds a path between two randomly chosen verti¬ 
ces and to calculate the average length of the paths found, for various random digraph 
models. 


4.2.42 Strong components. Run experiments to determine empirically the distribu¬ 
tion of the number of strong components in random digraphs of various types, by 
generating large numbers of digraphs and drawing a histogram. 


www.it-ebooks.info 


This page intentionally left blank 


www.it-ebooks.info 


4.3 MINIMUM SPANNING TREES 



an edge-weighted graph is a graph model where we associate weights or costs with each 
edge. Such graphs are natural models for many applications. In an airline map where 
edges represent flight routes, these weights might represent distances or fares. In an 
electric circuit where edges represent wires, the weights might represent the length of 
the wire, its cost, or the time that it takes a signal to propagate through it. Minimizing 
cost is naturally of interest in such situations. In this section, we consider undirected 
edge-weighted graph models and examine algorithms for one such problem: 


V- 


tinyEWG.txt 



An edge-weighted graph and its MST 


Minimum spanning tree. Given an undirected edge- 
weighted graph, find an MST. 

Definition. Recall that a spanning tree of a graph is a 
connected subgraph with no cycles that includes all 
the vertices. A minimum spanning tree (MST) of an 
edge-weighted graph is a spanning tree whose weight 
(the sum of the weights of its edges) is no larger than 
the weight of any other spanning tree. 


In this section, we examine two classical algorithms for 
computing MSTs: Prim’s algorithm and Kruskal’s algorithm. 
These algorithms are easy to understand and not difficult 
to implement. They are among the oldest and most well- 


application 

vertex 

edge 

circuit 

component 

wire 

airline 

airport 

flight route 

power 

distribution 

power plant 

transmission 

lines 

image 

analysis 

feature 

proximity 

relationship 


Typical MST applications 


known algorithms in this book, and they also take 
good advantage of modern data structures. Since 
MSTs have numerous important applications, al¬ 
gorithms to solve the problem have been studied at 
least since the 1920s, at first in the context of power 
distribution networks, later in the context of tele¬ 
phone networks. MST algorithms are now impor¬ 
tant in the design of many types of networks (com¬ 
munication, electrical, hydraulic, computer, road, 
rail, air, and many others) and also in the study of 
biological, chemical, and physical networks that are 
found in nature. 
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Assumptions. Various anomalous situations, which are generally easy to handle, can 
arise when computing minimum spanning trees. To streamline the presentation, we 


adopt the following conventions: 

■ The graph is connected. The spanning-tree condition in 
our definition implies that the graph must be connected 
for an MST to exist. Another way to pose the problem, 
recalling basic properties of trees from section 4 . 1 , is 
to find a minimal-weight set of V— 1 edges that connect 
the graph. If a graph is not connected, we can adapt our 
algorithms to compute the MSTs of each of its con¬ 
nected components, collectively known as a minimum 
spanning forest (see exercise 4 . 3 . 22 ). 

■ The edge weights are not necessarily distances. Geometric 
intuition is sometimes beneficial in understanding al¬ 
gorithms, so we use examples where vertices are points 
in the plane and weights are distances, such as the graph 
on the facing page. But it is important to remember that 
the weights might represent time or cost or an entirely 
different variable and do not need to be proportional to 
a distance at all. 


no MST if graph is not connected 



can independently compute 
MSTs of components 


weights need not be 
proportional to distance 



4 6 

5 6 
1 5 
0 4 
1 6 
0 2 
1 2 

1 3 

2 6 


0.61 

0.62 

0.88 

0.11 

0.35 

0.6 

0.10 

0.22 


0.62 

0.88 

0.02 

0.64 

0.90 

0.22 

0.50 

0.97 

0.17 


■ The edge weights may be zero or negative. If the edge 
weights are all positive, it suffices to define the MST as 
the subgraph with minimal total weight that connects 
all the vertices, as such a subgraph must form a span¬ 
ning tree. The spanning-tree condition in the definition 
is included so that it applies for graphs that may have 
zero negative edge weights. 

■ The edge weights are all different. If edges can have equal 
weights, the minimum spanning tree may not be unique 
(see exercise 4 . 3 . 2 ). The possibility of multiple MSTs 
complicates the correctness proofs of some of our algo¬ 
rithms, so we rule out that possibility in the presenta¬ 
tion. It turns out that this assumption is not restrictive 
because our algorithms work without modification in 
the presence of equal weights. 

In summary, we assume throughout the presentation that our 
job is to find the MST of a connected edge-weighted graph 
with arbitrary (but distinct) weights. 


weights can be 0 or negative 



4 

6 

0.62 

5 

6 

0.88 

1 

5 

0.02 

0 

4 

-0.99 

1 

6 

0 

0 

2 

0.22 

1 

2 

0.50 

1 

3 

0.97 

2 

6 

0.17 


MST may not be unique 
when weights have equal values 


^ © 

(1) f 13 0.50 

X_A 2 4 i-oo 

(^M 4 ) 3 4 0.50 


1 2 1-00 

(T/ 1 3 0.50 

©■Kl) 3 4 0.50 


Various MST anomalies 
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Underlying principles To begin, we recall from sec¬ 
tion 4.1 two of the defining properties of a tree: 

■ Adding an edge that connects two vertices in a tree 
creates a unique cycle. 

■ Removing an edge from a tree breaks it into two 
separate subtrees. 

These properties are the basis for proving a fundamental 
property of MSTs that leads to the MST algorithms that we 
consider in this section. 

Cut property. This property, which we refer to as the cut 
property, has to do with identifying edges that must be in the 
MST of a given edge-weighted graph, by dividing vertices 
into two sets and examining edges that cross the division. 


adding an edge 




removing an edge 
breaks tree into two parts 


Basic properties of a tree 


Definition. A cut of a graph is a partition of its vertices into two nonempty disjoint 
sets. A crossing edge of a cut is an edge that connects a vertex in one set with a vertex 
in the other. 


Typically, we specify a cut by specifying a set of vertices, leaving implicit the assumption 
that the cut comprises the given vertex set and its complement, so that a crossing edge 
is an edge from a vertex in the set to a vertex not in the set. In figures, we draw vertices 
on one side of the cut in gray and vertices on the other side in white. 

crossing edges separating 
gray from white vertices 
are drawn in red 

rC 

minimum-weight crossing edge 
must be in the MST 

Cut property 


Proposition J. (Cut property) Given any cut in an edge- 
weighted graph, the crossing edge of minimum weight is in 
the MST of the graph. 

Proof: Let e be the crossing edge of minimum weight and 
let T be the MST. The proof is by contradiction: Suppose 
that T does not contain e. Now consider the graph formed 
by adding e to T. This graph has a cycle that contains e, and 
that cycle must contain at least one other crossing edge— 
say,/, which has higher weight than e (since e is minimal and 
all edge weights are different). We can get a spanning tree of 
strictly lower weight by deleting / and adding e, contradict¬ 
ing the assumed minimality of T. 


www.it-ebooks.info 





4.3 ■ Minimum Spanning Trees 


607 


Under our assumption that edge weights 
are distinct, every connected graph has a 
unique MST (see exercise 4 . 3 . 3 ); and the 
cut property says that the shortest crossing 
edge for every cut must be in the MST. 

The figure to the left of proposition j il¬ 
lustrates the cut property. Note that there is 
no requirement that the minimal edge be the only MST edge connect¬ 
ing the two sets; indeed, for typical cuts there are several MST edges 
that connect a vertex in one set with a vertex in the other, as illustrated 
in the figure above. 



Greedy algorithm. The cut property is the basis for the algorithms 
that we consider for the MST problem. Specifically, they are special 
cases of a general paradigm known as the greedy algorithm: apply the 
cut property to accept an edge as an MST edge, continuing until find¬ 
ing all of the MST edges. Our algorithms differ in their approaches 
to maintaining cuts and identifying the crossing edge of minimum 
weight, but are special cases of the following: 


Proposition K. (Greedy MST algorithm) The following method 
colors black all edges in the the MST of any connected edge- 
weighted graph with V vertices: starting with all edges colored 
gray, find a cut with no black edges, color its minimum-weight 
edge black, and continue until V— 1 edges have been colored black. 

Proof: For simplicity, we assume in the discussion that the edge 
weights are all different, though the proposition is still true when 
that is not the case (see exercise 4 . 3 . 5 ). By the cut property, any 
edge that is colored black is in the MST. If fewer than V— 1 edges 
are black, a cut with no black edges exists (recall that we assume 
the graph to be connected). Once V—\ edges are black, the black 
edges form a spanning tree. 


The diagram at right is a typical trace of the greedy algorithm. Each 
drawing depicts a cut and identifies the minimum-weight edge in the 
cut (thick red) that is added to the MST by the algorithm. 




im. 



A. 



Greedy MST algorithm 
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Edge-weighted graph data type How should we represent edge-weighted 
graphs? Perhaps the simplest way to proceed is to extend the basic graph representa¬ 
tions from section 4 . 1 : in the adjacency-matrix representation, the matrix can contain 
edge weights rather than boolean values; in the adjacency-lists representation, we can 
define a node that contains both a vertex and a weight field to put in the adjacency lists. 
(As usual, we focus on sparse graphs and leave the adjacency-matrix representation for 
exercises.) This classic approach is appealing, but we will use a different method that is 
not much more complicated, will make our programs useful in more general settings, 
and needs a slightly more general API, which allows us to process Edge objects: 

public class Edge implements Comparable<Edge> 



Edge(int v, int w, double weight) 

initializing constructor 

double 

weight() 

weight of this edge 

int 

either0 

either of this edges vertices 

int 

otherfint v) 

the other vertex 

int 

compareTo(Edge that) 

compare this edge to e 

String 

toString() 

string representation 


API for a weighted edge 


The ei ther() and other O methods for accessing the edge’s vertices maybe a bit puz¬ 
zling at first—the need for them will become plain when we examine client code. You can 
find an implementation of Edge on page 610. It is the basis for this EdgeWei ghtedGraph 
API, which refers to Edge objects in a natural manner: 


public class 

EdgeWeightedGraph 



EdgeWeightedGraph(int V) 

create an empty V-vertex graph 


EdgeWeightedGraphfln in) 

read graph from input stream 

int 

V() 

number of vertices 

int 

E() 

number of edges 

voi d 

addEdgefEdge e) 

add edge e to this graph 

Iterable<Edge> 

adj (int v) 

edges incident to v 

Iterable<Edge> 

edgesO 

all of this graph’s edges 

String 

toString() 

string representation 


API for an edge-weighted graph 
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This API is very similar to the API for Graph (page 522). The two 
important differences are that it is based on Edge and that it adds the edges () method 
at right, which provides clients with the ability to iterate through to all the graph’s edges 
(ignoring any self-loops). The rest of the implementation of EdgeWeightedGraph on 
page 611 is quite similar to the unweighted undi¬ 
rected graph implementation of section 4 . 1 , 
but instead of the adjacency lists of integers 
used in Graph, it uses adjacency lists of Edge 
objects. 

The figure at the bottom of this page shows 
the edge-weighted graph representation that 
EdgeWeightedGraph builds from the sample 
file ti nyEWG. txt, showing the contents of each 
Bag as a linked list to reflect the standard imple¬ 
mentation of section 1 . 3 . To reduce clutter in the figure, we show each Edge as a pair 
of i nt values and a doubl e value. The actual data structure is a linked list of links to 
objects containing those values. In particular, although there are two references to each 
Edge (one in the list for each vertex), there is only one Edge object corresponding to 
each graph edge. In the figure, the edges appear in each list in reverse order of the order 
they are processed, because of the stack-like nature of the standard linked-list imple¬ 
mentation. As in Graph, by using a Bag we are making clear that our client code makes 
no assumptions about the order of objects in the lists. 


public Iterable<Edge> edges() 

{ 

Bag<Edge> b = new Bag<Edge>(); 
for (int v = 0; v < V; v++) 
for (Edge e : adj[v]) 

if (e.other(v) > v) b.add(e); 
return b; 


Gathering all the edges in an edge-weighted graph 
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Weighted edge data type 


public class Edge implements Comparable<Edge> 

{ 

private final int v; // one vertex 

private final int w; // the other vertex 

private final double weight; // edge weight 

public Edge(int v, int w, double weight) 

{ 

this.v = v; 
this.w = w; 
this.weight = weight; 

} 

public double weightO 
{ return weight; } 

public int either() 

{ return v; } 


public int other(int vertex) 

{ 

if (vertex == v) return w; 

else if (vertex == w) return v; 

else throw new RuntimeException("Inconsistent edge"); 

} 


public int 

{ 

if 

else if 
el se 


compareTo(Edge that) 


(this. weightO 
(this.weight() 


< that.weightO) 
> that.weightO) 


return -1 
return +1 
return 0 


public String toStringO 

{ return String.format("%d-%d %.2f", v, w, weight); } 


} 


This data type provides the methods eitherO and other() so that such clients can use other(v) to 
find the other vertex when it knows v. When neither vertex is known, our clients use the idiomatic 
codeint v = e.eitherO, w = e. other (v) ; to access an Edge e’s two vertices. 
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Edge-weighted graph data type 


public class EdgeWeightedGraph 

{ 

private final int V; // number of vertices 

private int E; // number of edges 

private Bag<Edge>[] adj; // adjacency lists 

public EdgeWeightedGraph(int V) 

{ 

this.V = V; 
this.E = 0; 

adj = (Bag<Edge>[]) new Bag[V]; 
for (int v = 0; v < V; v++) 
adj[v] = new Bag<Edge>(); 

} 

public EdgeWeightedGraph(In in) 

// See Exercise 4.3.9. 

public int V() { return V; } 
public int E() { return E; } 

public void addEdge(Edge e) 

{ 

int v = e.eitherO, w = e.other(v); 

adj[v].add(e) ; 
adj[w].add(e); 

E++; 

} 

public Iterable<Edge> adj(int v) 

{ return adj[v]; } 

public Iterable<Edge> edgesO 

// See page 609. 

} 


This implementation maintains a vertex-indexed array of lists of edges. As with Graph (see page 526), 
every edge appears twice: if an edge connects v and w, it appears both in v’s list and in w’s list. The 
edges () method puts all the edges in a Bag (see page 609). The toStri ng () implementation is left as 
an exercise. 
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Comparing edges by weight. The API specifies that the Edge class must implement the 
Comparabl e interface and include a compareToO implementation. The natural order¬ 
ing for edges in an edge-weighted graph is by weight. Accordingly, the implementation 
of compareToO is straightforward. 

Parallel edges. As with our undirected-graph implementations, we allow paral¬ 
lel edges. Alternatively, we could develop a more complicated implementation of 
EdgeWei ghtedGraph that disallows them, perhaps keeping the minimum-weight edge 
from a set of parallel edges. 

Self-loops. We allow self-loops. However, our edges () implementation in 
EdgeWei ghtedGraph does not include self-loops even though they might be present 
in the input or in the data structure. This omission has no effect on our MST algo¬ 
rithms because no MST contains a self-loop. When working with an application where 
self-loops are significant, you may need to modify our code as appropriate for the 
application. 

our choice to use explicit Edge objects leads to clear and compact client code, as you 
will see. It carries a small price: each adjacency-list node has a reference to an Edge ob¬ 
ject, with redundant information (all the nodes on v’s adjacency list have a v). We also 
pay object overhead cost. Although we have only one copy of each Edge, we do have two 
references to each Edge object. An alternative and widely used approach is to keep two 
list nodes corresponding to each edge, just as in Graph, each with a vertex and the edge 
weight in each list node. This alternative also carries a price—two nodes, including two 
copies of the weight for each edge. 
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MST API and test client As usual, for graph processing, we define an API where 
the constructor takes an edge-weighted graph as argument and supports client query 
methods that return the MST and its weight. How should we represent the MST itself? 
The MST of a graph G is a subgraph of G that is also a tree, so we have numerous op¬ 
tions. Chief among them are 

■ A list of edges 

■ An edge-weighted graph 

■ A vertex-indexed array with parent links 

To give clients and our implementations as much flexibility as possible in choosing 
among these alternatives for various applications, we adopt the following API: 


public class MST 

MST(EdgeWeightedCraph G) 
Iterable<Edge> edgesO 
double weightO 

API for MST implementations 


constructor 

all of the MST edges 

weight of MST 


Test client. As usual, we create sample graphs and develop a test client for use in test¬ 
ing our implementations. A sample client is shown below. It reads edges from the input 
stream, builds an edge-weighted graph, computes the MST of that graph, prints the 
MST edges, and prints the total weight of the MST. 


public static void main(String[] args) 
{ 

In in = new ln(args[0]); 

EdgeWeightedGraph G; 

G = new EdgeWeightedGraph(in); 

MST mst = new MST(G); 
for (Edge e : mst.edgesO) 
StdOut.println(e) ; 
StdOut.println(mst.weightO) I 


MST test client 
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Test data. You can find the file tinyEWG.txt on the booksite, which de¬ 


fines the small sample graph on page 604 that we use for detailed traces of MST 
algorithms. You can also find on the booksite the file mediumEWG.txt, which defines 
the weighted graph with 250 vertices that is drawn on bottom of the the facing page. 
It is an example of a Euclidean graph, whose vertices are points in the plane and whose 


edges are lines connecting them with weights equal to their Euclidean distances. Such 


graphs are useful for gaining insight into the behavior of MST algorithms, and they 


also model many of the typical practical problems we 
have mentioned, such as road maps or electric circuits. 
You can also find on the booksite is a larger example 
largeEWG.txt that defines a Euclidean graph with 
1 million vertices. Our goal is to be able to find the 
MST of such a graph in a reasonable amount of time. 


% more tinyEWG.txt 
8 16 
4 5 .35 

4 7 .37 

5 7 .28 
0 7 .16 

1 5 .32 
0 4 .38 

2 3 .17 
1 7 .19 
0 2 .26 
1 2 .36 

1 3 .29 

2 7 .34 

6 2 .40 

3 6 .52 
6 0 .58 
6 4 .93 

% java MST tinyEWG.txt 
0-7 0.16 

1- 7 0.19 
0-2 0.26 

2- 3 0.17 

5- 7 0.28 
4-5 0.35 

6- 2 0.40 
1.81 
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% more mediumEWG.txt 
250 1275 
244 246 0.11712 
239 240 0.10616 
238 245 0.06142 
235 238 0.07048 
233 240 0.07634 
232 248 0.10223 
231 248 0.10699 
229 249 0.10098 
228 241 0.01473 
226 231 0.07638 
... [1263 more edges ] 

% java MST mediumEWG.txt 
0 225 0.02383 
49 225 0.03314 
44 49 0.02107 

44 204 0.01774 
49 97 0.03121 

202 204 0.04207 
176 202 0.04299 
176 191 0.02089 
68 176 0.04396 
58 68 0.04795 

. . . [239 more edges ] 
10.46351 



graph 


MST 


A 250-node Euclidean graph (with 1,273 edges) and its MST 
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ineligible 
edge (gray) 


Prim’s algorithm Our first MST method, known as Prim’s algorithm, is to attach 
a new edge to a single growing tree at each step. Start with any vertex as a single-ver¬ 
tex tree; then add V~ 1 edges to it, always taking next (coloring black) the minimum- 
weight edge that connects a vertex on the tree to a vertex not yet 
on the tree (a crossing edge for the cut defined by tree vertices). 


crossing edge ( 

L 



minimum-weight 
crossing edge 
must he on MST 


Proposition L. Prim’s algorithm computes the MST of any 
connected edge-weighted graph. 

Proof: Immediate from proposition k. The growing tree 
defines a cut with no black edges; the algorithm takes the 
crossing edge of minimal weight, so it is successively coloring 
edges black in accordance with the greedy algorithm. 


Prim's MST algorithm 


The one-sentence description of Prim’s algorithm just given 
leaves unanswered a key question: How do we (efficiently) find 
the crossing edge of minimal weight? Several methods have been proposed—we will 
discuss some of them after we have developed a full solution based on a particularly 
simple approach. 


Data structures. We implement Prim’s algorithm with the aid of a few simple and 
familiar data structures. In particular, we represent the vertices on the tree, the edges on 
the tree, and the crossing edges, as follows: 

■ Vertices on the tree:~We use a vertex-indexed boolean array marked [], where 
marked [v] is true if v is on the tree. 

■ Edges on the tree: We use one of two data structures: a queue mst to collect MST 
edges or a vertex-indexed array edgeTo [] of Edge objects, where edgeTo [v] is 
the Edge that connects v to the tree. 

■ Crossing edges : We use a Mi nPQ<Edge> priority queue that compares edges by 
weight (see page 610). 

These data structures allow us to directly answer the basic question “Which is the min¬ 
imal-weight crossing edge?” 

Maintaining the set of crossing edges. Each time that we add an edge to the tree, we 
also add a vertex to the tree. To maintain the set of crossing edges, we need to add to 
the priority queue all edges from that vertex to any non-tree vertex (using marked [] to 
identify such edges). But we must do more: any edge connecting the vertex just added 
to a tree vertex that is already on the priority queue now becomes ineligible (it is no 
longer a crossing edge because it connects two tree vertices). An eager implementation 
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of Prim’s algorithm would remove such edges from 
the priority queue; we first consider a simpler lazy 
implementation of the algorithm where we leave 
such edges on the priority queue, deferring the eli¬ 
gibility test to when we remove them. 

The figure at right is a trace for our small sam¬ 
ple graph ti nyEWG. txt. Each drawing depicts the 
graph and the priority queue just after a vertex is 
visited (added to the tree and the edges in its ad¬ 
jacency list processed). The contents of the prior¬ 
ity queue are shown in order on the side, with new 
edges marked with asterisks. The algorithm builds 
the MST as follows: 

■ Adds 0 to the MST and all edges in its adja¬ 
cency list to the priority queue. 

■ Adds 7 and 0-7 to the MST and all edges in 
its adjacency list to the priority queue. 

■ Adds 1 and 1-7 to the MST and all edges in 
its adjacency list to the priority queue. 

■ Adds 2 and 0-2 to the MST and edges 2-3 
and 6-2 to the priority queue. Edges 2-7 and 
1-2 become ineligible. 

■ Adds 3 and 2-3 to the MST and edge 3-6 to 
the priority queue. Edge 1-3 becomes ineli¬ 
gible. 

■ Removes ineligible edges 1-3,1-5, and 2-7 
from the priority queue. 

■ Adds 5 and 5-7 to the MST and edge 4-5 to 
the priority queue. Edge 1-5 becomes ineli¬ 
gible. 

■ Adds 4 and 4 - 5 to the MST and edge 6-4 
to the priority queue. Edges 4-7 and 0-4 
become ineligible. 

■ Removes ineligible edges 1-2, 4-7, and 0-4 
from the priority queue. 

■ Adds 6 and 6-2 to the MST. The other edges 
incident to 6 become ineligible. 


* marks new 
entries 



* 1-7 0.19 
0-2 0.26 

* 5-7 0.28 

* 2-7 0.34 

* 4-7 0.37 
0-4 0.38 
6-0 0.58 


* 2-3 0.17 

5 - 7 0.28 
1-3 0.29 

1 - 5 0.32 

2 - 7 0.34 

1-2 0.36 

4-7 0.37 
0-4 0.38 

* 6-2 0.40 

6 - 0 0.58 


1-3 0.29 

1 - 5 0.32 

2 - 7 0.34 

* 4-5 0.35 
1-2 0.36 

4-7 0.37 
0-4 0.38 
6-2 0.40 

3 - 6 0.52 
6-0 0.58 


3-6 0.52 
6-0 0.58 
6-4 0.93 





0-2 0.26 
5-7 0.28 
1-3 0.29 

1 - 5 0.32 

2 - 7 0.34 
1-2 0.36 
4-7 0.37 
0-4 0.38 
0-6 0.58 





1-2 0.36 
4-7 0.37 
0-4 0.38 

6-2 0.40 
3-6 0.52 
6-0 0.58 
6-4 0.93 


Trace of Prim's algorithm (lazy version) 
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After having added V vertices (and V~ 1 edges), the MST is complete. The remaining 
edges on the priority queue are ineligible, so we need not examine them again. 

Implementation. With these preparations, implementing Prim’s algorithm is straight¬ 
forward, as shown in the implementation LazyPri mMST on the facing page. As with our 
depth-first search and breadth-first search implementations in the previous two sec¬ 
tions, it computes the MST in the constructor so that client methods can learn proper¬ 
ties of the MST with query methods. We use a private method vi si t() that puts a ver¬ 
tex on the tree, by marking it as visited and then putting all of its incident edges that are 
not ineligible onto the priority queue, thus ensuring that the priority queue contains 
the crossing edges from tree vertices to non-tree vertices (perhaps also some ineligible 
edges). The inner loop is a rendition in code of the one-sentence description of the al¬ 
gorithm: we take an edge from the priority queue and (if it is not ineligible) add it to the 
tree, and also add to the tree the new vertex that it leads to, updating the set of crossing 
edges by calling vi si t() with that vertex as argument. The wei ght() method requires 
iterating through the tree edges to add up the edge weights (lazy approach) or keeping 
a running total in an instance variable (eager approach) and is left as exercise 4 . 3 . 31 . 

Running time. How fast is Prim’s algorithm? This question is not difficult to answer, 
given our knowledge of the behavior characteristics of priority queues: 

Proposition M. The lazy version of Prim’s algorithm uses space proportional to E 
and time proportional to E log E (in the worst case) to compute the MST of a con¬ 
nected edge-weighted graph with E edges and V vertices. 

Proof: The bottleneck in the algorithm is the number of edge-weight comparisons 
in the priority-queue methods i nsertO and delMi n(). The number of edges on 
the priority queue is at most E, which gives the space bound. In the worst case, 
the cost of an insertion is ~lg E and the cost to delete the minimum is ~2lg E (see 
proposition o in chapter 2 ). Since at most E edges are inserted and at most E are 
deleted, the time bound follows. 


In practice, the upper bound on the running time is a bit conservative because the 
number of edges on the priority queue is typically much less than E. The existence of 
such a simple, efficient, and useful algorithm for such a challenging task is quite re¬ 
markable. Next, we briefly discuss some improvements. As usual, detailed evaluation of 
such improvements in performance-critical applications is a job for experts. 
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Lazy version of Prim's MST algorithm 


public class LazyPrimMST 

{ 

private booleanf] marked; // MST vertices 

private Queue<Edge> mst; // MST edges 

private MinPQ<Edge> pg; // crossing (and ineligible) edges 

public LazyPrimMST(EdgeWeightedCraph G) 

{ 

pq = new MinPQ<Edge>(); 
marked = new boolean[G.V()]; 
mst = new Queue<Edge>(); 

visit(G, 0); // assumes G is connected (see Exercise 4.3.22) 

while (!pq.isEmptyO) 

{ 

Edge e = pq.delMinO; // Get lowest-weight 

int v = e.eitherO, w = e.other(v); // edge from pq. 
if (marked[v] && marked[w]) continue; // Skip if ineligible. 
mst.enqueue(e); // Add edge to tree, 

if (!marked[v]) visit(G, v); // Add vertex to tree 

if (!marked[w]) visit(G, w); // (either v or w). 

} 

} 

private void visit(EdgeWeightedGraph G, int v) 

{ // Mark v and add to pq all edges from v to unmarked vertices. 

marked[v] = true; 
for (Edge e : G.adj(v)) 

if (!marked[e.other(v)]) pq.insert(e); 

} 

public Iterable<Edge> edges() 

{ return mst; } 

public double weight() // See Exercise 4.3.31. 

} 


This implementation of Prim’s algorithm uses a priority queue to hold crossing edges, a vertex-in¬ 
dexed arrays to mark tree vertices, and a queue to hold MST edges. This implementation is a lazy 
approach where we leave ineligible edges in the priority queue. 
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Eager version of Prim’s algorithm To improve the LazyPri mMST, we might try 
to delete ineligible edges from the priority queue, so that the priority queue contains 
only the crossing edges between tree vertices and non-tree vertices. But we can elimi¬ 
nate even more edges. The key is to note that our only interest is in the minimal edge 
from each non-tree vertex to a tree vertex. When we add a vertex 
v to the tree, the only possible change with respect to each non¬ 
tree vertex w is that adding v brings w closer than before to the 
tree. In short, we do not need to keep on the priority queue all 
of the edges from w to tree vertices—we just need to keep track 
of the minimum-weight edge and check whether the addition 
of v to the tree necessitates that we update that minimum (be¬ 
cause of an edge v-w that has lower weight), which we can do 
as we process each edge in v’s adjacency list. In other words, we 
maintain on the priority queue just one edge for each non-tree vertex w: the shortest 
edge that connects it to the tree. Any longer edge connecting w to the tree will become 
ineligible at some point, so there is no need to keep it on the priority queue. 

PrimMST (algorithm 4.7 on page 622) implements Prim’s algorithm using our in¬ 
dex priority queue data type from section 2.4 (see page 320). It replaces the data 
structures marked [] and ms t [] in LazyPri mMST by two vertex-indexed arrays edgeTo [] 
and di stTo [] , which have the following properties: 

■ If v is not on the tree but has at least one edge connecting it to the tree, then 
edgeTo [v] is the shortest edge connecting v to the tree, and di stTo [v] is the 
weight of that edge. 

■ All such vertices v are maintained on the index priority queue, as an index v as¬ 
sociated with the weight of edgeTo [v]. 

The key implications of these properties is that the minimum key on the priority queue 
is the weight of the minimal-weight crossing edge, and its associated vertex v is the next 
to add to the tree. The marked [] array is not needed, since the condition ! marked [w] 
is equivalent to the condition that di stTo [w] is infinite (and that edgeTo [w] is null). 
To maintain the data structures, Pri mMST takes an edge v from the priority queue, then 
checks each edge v-w on its adjacency list. If w is marked, the edge is ineligible; if it is not 
on the priority queue or its weight is lower than the current best-known edgeTo [w], 
the code updates the data structures to establish v-w as the best-known way to connect 
v to the tree. 

The figure on the facing page is a trace of PrimMST for our small sample graph 
tinyEWG.txt. The contents of the edgeTo [] and distTo[] arrays are depicted after 
each vertex is added to the MST, color-coded to depict the MST vertices (index in black), 
the non-MST vertices (index in gray), the MST edges (in black), and the priority-queue 


connecting v 
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index/value pairs (in red). In the drawings, 
the shortest edge connecting each non-MST 
vertex to an MST vertex is drawn in red. The 
algorithm adds edges to the MST in the same 
order as the lazy version; the difference is in 
the priority-queue operations. It builds the 
MST as follows: 

■ Adds 0 to the MST and all edges in its 
adjacency list to the priority queue, 
since each such edge is the best (only) 
known connection between a tree ver¬ 
tex and a non-tree vertex. 

■ Adds 7 and 0-7 to the MST and 1-7 
and 5-7 to the priority queue. Edges 
4-7 and 2-7 do not affect the priority 
queue because their weights are not less 
than the weights of the known connec¬ 
tions from the MST to 4 and 2, respec¬ 
tively. 

■ Adds 1 and 1-7 to the MST and 1-3 to 
the priority queue. 

■ Adds 2 and 0-2 to the MST, replaces 
0-6 with 2-6 as the shortest edge from 
a tree vertex to 6, and replaces 1- 3 with 
2 - 3 as the shortest edge from a tree 
vertex to 3. 

■ Adds 3 and 2 - 3 to the MST. 

■ Adds 5 and 5-7 to the MST and re- 



edgeTo[] distTo[] 
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3 2 - 0 . 17 -*— 

4 0-4 0.38 

5 5-7 0.28 

6 -2 0.40 

7 0-7 0.16 
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1 1-7 0.19 

2 0-2 0.26 

3 2-3 0.17 

4 0 - 0.38 

5 -7 0.28 

6 -2 0.40 

7 0-7 0.16 

0 

1 1-7 0.19 

2 0-2 0.26 

3 2-3 0.17 

4 4-5 0.35 

5 5-7 0.28 

6 -2 0.40 

7 0-7 0.16 


places 0-4 with 4-5 as the shortest edge 
from a tree vertex to 4. 

■ Adds 4 and 4 - 5 to the MST. 

■ Adds 6 and 6-2 to the MST. 

After having added V~l edges, the MST is 
complete and the priority queue is empty. 

AN ESSENTIALLY IDENTICAL ARGUMENT as in 

the proof of proposition m proves that the 
eager version of Prim’s algorithm finds the 
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ALGORITHM 4.7 Prim's MST algorithm (eager version) 


public class PrimMST 

{ 

private Edge[] edgeTo; 
private doublet] distTo; 
private boolean[] marked; 
private IndexMinPQ<Double> pq; 


// shortest edge from tree vertex 
// distTo[w] = edgeTo[w].weightC) 
// true if v on tree 
// eligible crossing edges 


public PrimMST(EdgeWeightedGraph G) 

{ 


edgeTo = new Edge[G.V()]; 

distTo = new double[G.V()]; 

marked = new boolean[G.V()]; 

for (int v = 0; v < G.V(); v++) 

distTo[v] = Double.POSITIVE_INFINITY; 
pq = new IndexMinPQ<Double>(G.V()); 


distTo[0] = 0.0; 

pq.insert(0, 0.0); // Initialize pq with 0, weight 0. 

while (! pq. i sEmptyO) 

visitCG, pq.delMinQ) ; // Add closest vertex to tree. 


private void visit(EdgeWeightedGraph G, int v) 

{ // Add v to tree; update data structures. 

marked[v] = true; 
for (Edge e : G.adj(v)) 

{ 


int w = e.other(v); 

if (marked[w]) continue; // v-w 

if (e.weightO < distTo[w]) 

{ // Edge e is new best connection 

edgeTo[w] = e; 
distTo[w] = e.weightO; 
if (pq.contains(w)) pq.change(w, 
else pq.insert(w, 


is ineligible, 
from tree to w. 

distTo[w]); 
distTo[w]); 


} 


} 


} 


public Iterable<Edge> edges() 
public double weightO 


// See Exercise 4.S.21. 
// See Exercise 4.3.31. 


This implementation of Prim’s algorithm keeps eligible crossing edges on an index priority queue. 
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MST of a connected edge-weighted graph in time proportional 
to E log V and extra space proportional to V (see page 623). For the 
huge sparse graphs that are typical in practice, there is no asymp¬ 
totic difference in the time bound (because lg E ~ lg V for sparse 
graphs); the space bound is a constant-factor (but significant) 
improvement. Further analysis and experimentation are best left 
for experts facing performance-critical applications, where many 
factors come into play, including the implementations of MinPQ 
and IndexMinPQ, the graph representation, properties of the ap¬ 
plication’s graph model, and so forth. As usual, such improvements 
need to be carefully considered, as the increased code complexity is 
only justified for applications where constant-factor performance 
gains are important, and might even be counterproductive on 
complex modern systems. 

Proposition N. The eager version of Prim’s algorithm uses ex¬ 
tra space proportional to V and time proportional toil log V 
(in the worst case) to compute the MST of a connected edge- 
weighted graph with E edges and V vertices. 

Proof: The number of edges on the priority queue is at most 
V, and there are three vertex-indexed arrays, which implies the 
space bound. The algorithm uses V insert operations, V delete 
the minimum operations, and (in the worst case) E change pri¬ 
ority operations. These counts, coupled with the fact that our 
heap-based implementation of the index priority queue imple¬ 
ments all these operations in time proportional to log V (see 
page 321), imply the time bound. 


The diagram at right shows Prim’s algorithm in operation on our 
250-vertex Euclidean graph medi umEWG. txt. It is a fascinating dy¬ 
namic process (see also exercise 4 . 3 . 27 ). Most often the tree grows 
by connecting a new vertex to the vertex just added. When reaching 
an area with no nearby non-tree vertices, the growth starts from 
another part of the tree. 
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Prim's algorithm (250 vertices) 
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{gray) 


0-7 0.16 
2-3 0.17 
1-7 0.19 
0-2 0.26 

5 - 7 0.28 
1-3 0.29 

1 - 5 0.32 

2 - 7 0.34 
4-5 0.35 
1-2 0.36 
4-7 0.37 
0-4 0.38 

6 - 2 0.40 

3 - 6 0.52 
6-0 0.58 
6-4 0.93 


© 



Trace of Kruskal's algorithm 


Kruskal’s algorithm The second MST algorithm 
that we consider in detail is to process the edges in order of 
their weight values (smallest to largest), taking for the MST 
(coloring black) each edge that does not form a cycle with 
edges previously added, stopping after adding V— 1 edges 
have been taken. The black edges form a forest of trees that 
evolves gradually into a single tree, the MST This method 
is known as Kruskal’s algorithm: 

Proposition O. Kruskal’s algorithm computes the 
MST of any edge-weighted connected graph. 

Proof: Immediate from proposition k. If the next 
edge to be considered does not form a cycle with black 
edges, it crosses a cut defined by the set of vertices 
connected to one of the edge’s vertices by tree edges 
(and its complement). Since the edge does not create a 
cycle, it is the only crossing edge seen so far, and since 
we consider the edges in sorted order, it is a crossing 
edge of minimum weight. Thus, the algorithm is suc¬ 
cessively taking a minimal-weight crossing edge, in ac¬ 
cordance with the greedy algorithm. 


Prim’s algorithm builds the MST one edge at a time, finding 
a new edge to attach to a single growing tree at each step. 
Kruskal’s algorithm also builds the MST one edge at a time; 
but, by contrast, it finds an edge that connects two trees in a 
forest of growing trees. We start with a degenerate forest of 
V single-vertex trees and perform the operation of combin¬ 
ing two trees (using the shortest edge possible) until there is 
just one tree left: the MST. 

The figure at left shows a step-by-step example of the 
operation of Kruskal’s algorithm on ti ny EWG. txt. The five 
lowest-weight edges in the graph are taken for the MST, 
then 1-3, 1-5, and 2-7 are determined to be ineligible be¬ 
fore 4-5 is taken for the MST, and finally 1-2, 4-7, and 0-4 
are determined to be ineligible and 6-2 is taken for the MST. 
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Kruskal’s algorithm is also not difficult to implement, given the basic algorithmic 
tools that we have considered in this book: we use a priority queue (section 2 . 4 ) to 
consider the edges in order by weight, a union-find data structure (section 1 . 5 ) to 
identify those that cause cycles, and a queue (section 1 . 3 ) to collect the MST edges. 
algorithm 4.8 is an implementation along these lines. Note that collecting the MST 
edges in a Queue means that when a client iterates through the edges it gets them in 
increasing order of their weight. The wei ght() method requires iterating through the 
queue to add the edge weights (or keeping a running total in an instance variable) and 
is left as an exercise (see exercise 4 . 3 . 31 ). 

Analyzing the running time of Kruskal’s algorithm is a simple matter because we 
know the running times of its basic operations. 

Proposition N (continued). Kruskal’s algorithm uses space proportional to E and 
time proportional to I: log E (in the worst case) to compute the MST of an edge- 
weighted connected graph with E edges and V vertices. 

Proof: The implementation uses the priority-queue constructor that initializes the 
priority queue with all the edges, at a cost of at most E compares (see section 2 . 4 ). 
After the priority queue is built, the argument is the same as for Prim’s algorithm. 
The number of edges on the priority queue is at most E, which gives the space 
bound, and the cost per operation is at most 2 lg E compares, which gives the time 
bound. Kruskal’s algorithm also performs up to E find() and V unionO opera¬ 
tions, but that cost does not contribute to the E log E order of growth of the total 
running time (see section 1 . 5 ). 


As with Prim’s algorithm the cost bound is conservative, since the algorithm terminates 
after finding the V— 1 MST edges. The order of growth of the actual cost is E + E 0 log E, 
where E 0 is the number of edges whose weight is less than the weight of the MST edge 
with the highest weight. Despite this advantage, Kmskal’s algorithm is generally slower 
than Prim’s algorithm because it has to do a connected () operation for each edge, in 
addition to the priority-queue operations that both algorithms do for each edge pro¬ 
cessed (see exercise 4 . 3 . 39 ). 
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Kruskal's algorithm (250 vertices) 


The figure at left illustrates the algorithm’s dynamic characteris¬ 
tics on the larger example medi umEWG. txt. The fact that the edges 
are added to the forest in order of their length is quite apparent. 
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ALGORITHM 4.8 Kruskal's MST algorithm 


public class KruskalMST 

{ 

private Queue<Edge> mst; 

public KruskalMST(EdgeWeightedGraph G) 

{ 

mst = new Queue<Edge>(); 

MinPQ<Edge> pg = new MinPQ<Edge>(G.edgesO) ; 

UF uf = new UF(G.VO) ; 

while (!pq.isEmptyO && mst.sizeO < G.V()-1) 

{ 

Edge e = pg.delMinO; // Get min weight edge on pq 

int v = e.eitherO, w = e.other(v); // and its vertices, 
if (uf.connected(v, w)) continue; // Ignore ineligible edges. 
uf.union(v, w); // Merge components, 

mst.enqueue(e); // Add edge to mst. 

} 

} 

public Iterable<Edge> edgesO 
{ return mst; } 

public double weightQ // See Exercise 4.S.31. 


This implementation of Kruskal’s algorithm uses a queue to hold MST edges, a priority queue to hold 
edges not yet examined, and a union-find data structure for identifying ineligible edges. The MST 
edges are returned to the client in increasing order of their weights. The wei ght() method is left as 
an exercise. 


% java KruskalMST tinyEWG.txt 
0-7 0.16 
2-3 0.17 
1-7 0.19 
0-2 0.26 

5- 7 0.28 
4-5 0.35 

6- 2 0.40 
1.81 
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Perspective The MST problem is one of the most heavily studied problems that 
we encounter in this book. Basic approaches to solving it were invented long before 
the development of modern data structures and modern techniques for analyzing the 
performance of algorithms, at a time when finding the MST of a graph that contained, 
say, thousands of edges was a daunting task. The MST algorithms that we have consid¬ 
ered differ from these old ones essentially in their use and implementation of modern 
algorithms and data structures for basic tasks, which (coupled with modern comput¬ 
ing power) makes it possible for us to compute MSTs with millions or even billions of 
edges. 


algorithm 


worst-case order of growth for 
V vertices and E edges 


space 


time 


lazy Prim 


eager Prim 
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Chazelle 


impossible ? 
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E+V logy 

very, very nearly, 
but not quite E 

El 


Historical notes. An MST imple¬ 
mentation for dense graphs (see ex¬ 
ercise 4 . 3 . 29 ) was first presented by 
R. Prim in 1961 and, independently, 
by E. W. Dijkstra soon thereafter. It 
is usually referred to as Prim’s algo¬ 
rithm, although Dijkstra’s presenta¬ 
tion was more general. But the basic 
idea was also presented by V. Jarnik 
in 1939, so some authors refer to the 
method as Jarnik’s algorithm, thus 
characterizing Prim’s (or Dijkstra’s) 
role as finding an efficient imple¬ 
mentation of the algorithm for dense 
graphs. As the priority-queue ADT 
came into use in the early 1970s, its 
application to finding MSTs of sparse 

graphs was straightforward; the fact that MSTs of sparse graphs could be computed in 
time proportional to E log E became widely known without attribution to any particu¬ 
lar researcher. In 1984, M. L. Fredman and R. E. Tarjan developed the Fibonacci heap 
data structure, which improves the theoretical bound on the order of growth of the 
running time of Prim’s algorithm to E+V log V. J. Kruskal presented his algorithm 
in 1956, but, again, the relevant ADT implementations were not carefully studied for 
many years. Other interesting historical notes are that Kruskal’s paper mentioned a ver¬ 
sion of Prim’s algorithm and that a 1926 (!) paper by O. Boruvka mentioned both ap¬ 
proaches. Boruvka’s paper addressed a power-distribution application and introduced 
yet another method that is easily implemented with modern data structures (see exer¬ 
cise 4 . 3.43 and exercise 4 . 3 . 44 ). The method was rediscovered by M. Sollin in 1961; 


Performance characteristics of MST algorithms 
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it later attracted attention as the basis for MST algorithms with efficient asymptotic 
performance and as the basis for parallel MST algorithms. 

A linear-time algorithm ? On the one hand, no theoretical results have been developed 
that deny the existence of an MST algorithm that is guaranteed to run in linear time 
for all graphs. On the other hand, the goal of developing algorithms for computing 
the MST of sparse graphs in linear time remains elusive. Since the 1970s the appli¬ 
cability of the union-find abstraction to Kruskal’s algorithm and the applicability of 
the priority-queue abstraction to Prim’s algorithm have been prime motivations for 
many researchers to seek better implementations of those ADTs. Many researchers have 
concentrated on finding efficient priority-queue implementations as the key to find¬ 
ing efficient MST algorithms for sparse graphs; many other researchers have studied 
variations of Boruvka’s algorithm as the basis for nearly linear-time MST algorithms 
for sparse graphs. Such research still holds the potential to lead us eventually to a prac¬ 
tical linear-time MST algorithm and has even shown the existence of a randomized 
linear-time algorithm. Also, researchers are getting quite close to the linear-time goal: 
B. Chazelle exhibited an algorithm in 1997 that certainly could never be distinguished 
from a linear-time algorithm in any conceivable practical situation (even though it is 
provably nonlinear), but is so complicated that no one would use it in practice. While 
the algorithms that have emerged from such research are generally quite complicated, 
simplified versions of some of them may yet be shown to be useful in practice. In the 
meantime, we can use the basic algorithms that we have considered here to compute 
the MST in linear time in most practical situations, perhaps paying an extra factor of 
log V for some sparse graphs. 

in summary, we can consider the MST problem to be “solved” for practical purposes. 
For most graphs, the cost of finding the MST is only slightly higher than the cost of 
extracting the graph’s edges. This rule holds except for huge graphs that are extremely 
sparse, but the available performance improvement over the best-known algorithms 
even in this case is a small constant factor, perhaps a factor of 10 at best. These conclu¬ 
sions are borne out for many graph models, and practitioners have been using Prim’s 
and Kruskal’s algorithms to find MSTs in huge graphs for decades. 
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| Q&A 


Q. Do Prim’s and Kruskal’s algorithms work for directed graphs? 

A. No, not at all. That is a more difficult graph-processing problem known as the mini¬ 
mum cost arborescence problem. 
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EXERCISES 

4 . 3.1 Prove that you can rescale the weights by adding a positive constant to all of 
them or by multiplying them all by a positive constant without affecting the MST. 

4 . 3.2 Draw all of the MSTs of the graph depicted at right (all edge weights are 
equal). 

4 . 3.3 Show that if a graph’s edges all have distinct weights, the MST is unique. 

4 . 3.4 Consider the assertion that an edge-weighted graph has a unique MST only if its 
edge weights are distinct. Give a proof or a counterexample. 

4 . 3.5 Show that the greedy algorithm is valid even when edge weights are not distinct. 

4 . 3.6 Give the MST of the weighted graph obtained by deleting vertex 7 from 
ti nyEWG. txt (see page 604). 

4 . 3.7 How would you find a maximum spanning tree of an edge-weighted graph? 

4 . 3.8 Prove the following, known as the cycle property: Given any cycle in an edge- 
weighted graph (all edge weights distinct), the edge of maximum weight in the cycle 
does not belong to the MST of the graph. 

4 . 3.9 Implement the constructor for EdgeWei ghtedGraph that reads a graph from the 
input stream, by suitably modifying the constructor from Graph (see page 526). 

4 . 3.10 Develop an EdgeWei ghtedGraph implementation for dense graphs that uses an 
adjacency-matrix (two-dimensional array of weights) representation. Disallow parallel 
edges. 

4 . 3.11 Determine the amount of memory used by EdgeWei ghtedGraph to represent a 
graph with V vertices and E edges, using the memory-cost model of section 1 . 4 . 

4 . 3.12 Suppose that a graph has distinct edge weights. Does its shortest edge have to 
belong to the MST? Can its longest edge belong to the MST? Does a min-weight edge 
on every cycle have to belong to the MST? Prove your answer to each question or give 
a counterexample. 

4 . 3.13 Give a counterexample that shows why the following strategy does not neces¬ 
sarily find the MST: ‘Start with any vertex as a single-vertex MST, then add V-l edges 
to it, always taking next a min-weight edge incident to the vertex most recently added 
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EXERCISES (continued) 


to the MST.’ 

4.3.14 Given an MST for an edge-weighted graph G, suppose that an edge in G that 
does not disconnect G is deleted. Describe how to find an MST of the new graph in time 
proportional to E. 

4.3.15 Given an MST for an edge-weighted graph G and a new edge e, describe how to 
find an MST of the new graph in time proportional to V. 

4.3.16 Given an MST for an edge-weighted graph G and a new edge e, write a program 
that determines the range of weights for which e is in an MST 

4.3.17 Implement toStringO for EdgeWeightedGraph. 

4.3.18 Give traces that show the process of computing the MST of the graph defined 
in exercise 4 . 3.6 with the lazy version of Prim’s algorithm, the eager version of Prim’s 
algorithm, and Kruskal’s algorithm. 

4.3.19 Suppose that you use a priority-queue implementation that maintains a sorted 
list. What would be the order of growth of the worst-case running time for Prim’s algo¬ 
rithm and for Kruskal’s algorithm for graphs with V vertices and E edges? When would 
this method be appropriate, if ever? Defend your answer. 

4.3.20 True or false: At any point during the execution of Kruskal’s algorithm, each 
vertex is closer to some vertex in its subtree than to any vertex not in its subtree. Prove 
your answer. 

4.3.21 Provide an implementation of edges () for Pri mMST (page 622). 

Solution: 

public Iterable<Edge> edgesf) 

{ 

Bag<Edge> mst = new Bag<Edge>(); 
for (int v = 1; v < edgeTo.length; v++) 
mst.add(edgeTo[v]); 
return mst; 

} 
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CREATIVE PROBLEMS 


4.3.22 Minimum spanning forest. Develop versions of Prim’s and Kruskal’s algorithms 
that compute the minimum spanning forest of an edge-weighted graph that is not nec¬ 
essarily connected. Use the connected-components API of section 4.1 and find MSTs 
in each component. 

4.3.23 Vyssotsky’s algorithm. Develop an implementation that computes the MST by 
applying the cycle property (see exercise 4 . 3 . 8 ) repeatedly: Add edges one at a time to 
a putative tree, deleting a maximum-weight edge on the cycle if one is formed. Note: 
This method has received less attention than the others that we consider because of the 
comparative difficulty of maintaining a data structure that supports efficient imple¬ 
mentation of the “delete the maximum-weight edge on the cycle” operation. 

4.3.24 Reverse-delete algorithm. Develop an implementation that computes the MST 
as follows: Start with a graph containing all of the edges. Then repeatedly go through 
the edges in decreasing order of weight. For each edge, check if deleting that edge will 
disconnect the graph; if not, delete it. Prove that this algorithm computes the MST. 
What is the order of growth of the number of edge-weight compares performed by your 
implementation? 

4.3.25 Worst-case generator. Develop a reasonable generator for edge-weighted graphs 
with V vertices and E edges such that the running time of the lazy version of Prim’s al¬ 
gorithm is nonlinear. Answer the same question for the eager version. 

4.3.26 Critical edges. An MST edge whose deletion from the graph would cause the 
MST weight to increase is called a critical edge. Show how to find all critical edges in a 
graph in time proportional to E log E. Note: This question assumes that edge weights 
are not necessarily distinct (otherwise all edges in the MST are critical). 

4.3.27 Animations. Write a client program that does dynamic graphical animations 
of MST algorithms. Run your program for medi umEWG. txt to produce images like the 
figures on page 621 and page 624. 

4.3.28 Space-efficient data structures. Develop an implementation of the lazy ver¬ 
sion of Prim’s algorithm that saves space by using lower-level data structures for 
EdgeWei ghtedGraph and for Mi nPQ instead of Bag and Edge. Estimate the amount of 
memory saved as a function of V and E, using the memory-cost model of section 1.4 
(see exercise 4 . 3 . 11 ). 
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CREATIVE PROBLEMS (continued) 


4.3.29 Dense graphs. Develop an implementation of Prim’s algorithm that uses an 
eager approach (but not a priority queue) and computes the MST using V 2 edge-weight 
comparisons. 

4.3.30 Euclidean weighted graphs. Modify your solution to exercise 4 . 1.37 to cre¬ 
ate an API Eucl i deanEdgeWei ghtedGraph for graphs whose vertices are points in the 
plane, so that you can work with graphical representations. 

4.3.31 MST weights. Develop implementations of weightO for LazyPrimMST, 
PrimMST, and KruskalMST, using a lazy strategy that iterates through the MST edges 
when the client calls weightO. Then develop alternate implementations that use an 
eager strategy that maintains a running total as the MST is computed. 

4.3.32 Specified set. Given a connected edge-weighted graph G and a specified set of 
edges S (having no cycles), describe a way to find a minimum-weight spanning tree of 
G that contains all the edges in S. 

4.3.33 Certification. Write an MST and EdgeWeightedGraph client checkO that uses 
the following cut optimality conditions implied by proposition j to verify that a pro¬ 
posed set of edges is in fact an MST: A set of edges is an MST if it is a spanning tree and 
every edge is a minimum-weight edge in the cut defined by removing that edge from 
the tree. What is the order of growth of the running time of your method? 
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EXPERIMENTS 


4.3.34 Random sparse edge-weighted graphs. Write a random-sparse-edge-weighted- 
graph generator based on your solution to exercise 4 . 1 . 41 . To assign edge weights, 
define a random-edge-weighted digraph ADT and write two implementations: one that 
generates uniformly distributed weights, another that generates weights according to a 
Gaussian distribution. Write client programs to generate sparse random edge-weighted 
graphs for both weight distributions with a well-chosen set of values of V and E so that 
you can use them to run empirical tests on graphs drawn from various distributions of 
edge weights. 

4.3.35 Random Euclidean edge-weighted graphs. Modify your solution to exercise 

4 . 1.42 to assign the distance between vertices as each edge’s weight. 

4.3.36 Random grid edge-weighted graphs. Modify your your solution to exercise 

4 . 1.43 to assign a random weight (between 0 and 1 ) to each edge. 

4.3.37 Real edge-weighted graphs. Find a large weighted graph somewhere online— 
perhaps a map with distances, telephone connections with costs, or an airline rate 
schedule. Write a program RandomReal EdgeWeightedGraph that builds a weighted 
graph by choosing V vertices at random and E weighted edges at random from the 
subgraph induced by those vertices. 

Testing all algorithms and studying all parameters against all graph models is unrealistic. 
For each problem listed below, write a client that addresses the problem for any given input 
graph, then choose among the generators above to run experiments for that graph model. 
Use your judgment in selecting experiments, perhaps in response to results of previous 
experiments. Write a narrative explaining your results and any conclusions that might be 
drawn. 


4.3.38 Cost of laziness. Run empirical studies to compare the performance of the lazy 
version of Prim’s algorithm with the eager version, for various types of graphs. 

4.3.39 Prim versus Kruskal. Run empirical studies to compare the performance of the 
lazy and eager versions of Prim’s algorithm with Kruskal’s algorithm. 

4.3.40 Reduced overhead. Run empirical studies to determine the effect of using 
primitive types instead of Edge values in EdgeWeightedGraph, as described in exer¬ 
cise 4 . 3 . 28 . 
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EXPERIMENTS (continued) 


4.3.41 Longest MST edge. Run empirical studies to analyze the length of the longest 
edge in the MST and the number of graph edges that are not longer than that one. 

4.3.42 Partitioning. Develop an implementation based on integrating Kruskal’s al¬ 
gorithm with quicksort partitioning (instead of using a priority queue) so as to check 
MST membership of each edge as soon as all smaller edges have been checked. 

4.3.43 Boruvka’s algorithm. Develop an implementation of Boruvka’s algorithm: 
Build an MST by adding edges to a growing forest of trees, as in Kruskal’s algorithm, 
but in stages. At each stage, find the minimum-weight edge that connects each tree to a 
different one, then add all such edges to the MST. Assume that the edge weights are all 
different, to avoid cycles. Hint: Maintain in a vertex-indexed array to identify the edge 
that connects each component to its nearest neighbor, and use the union-find data 
structure. 

4.3.44 Improved Boruvka. Develop an implementation of Boruvka’s algorithm that 
uses doubly-linked circular lists to represent MST subtrees so that subtrees can be 
merged and renamed in time proportional to E during each stage (and the union-find 
ADT is therefore not needed). 

4.3.45 External MST. Describe how you would find the MST of a graph so large that 
only V edges can fit into main memory at once. 

4.3.46 Johnson’s algorithm. Develop a priority-queue implementation that uses a d- 
way heap (see exercise 2 . 4 . 41 ). Find the best value of d for various weighted graph 
models. 
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4.4 SHORTEST PATHS 



perhaps the most intuitive graph-processing problem is one that you encounter 
regularly, when using a map application or a navigation system to get directions from 
one place to another. A graph model is immediate: vertices correspond to intersections 
and edges correspond to roads, with weights on the edges that model the cost, perhaps 
distance or travel time. The possibility of one-way roads means that we will need to 
consider edge-weighted digraphs. In this model, the problem is easy to formulate: 

Find the lowest-cost way to get from one vertex to another. 


Beyond direct applications of this sort, the shortest-paths model is appropriate for a 
range of other problems, some of which do not seem to be at all related to graph pro¬ 
cessing. As one example, we shall consider the arbitrage problem from computational 
finance at the end of this section. 


We adopt a general model where 
we work with edge-weighted digraphs 
(combining the models of section 

application 

vertex 

edge 

map 

intersection 

road 

4.2 and section 4.3). In section 

network 

router 

connection 

4.2 we wished to know whether it is 
possible to get from one vertex to an- 

schedule 

job 

precedence constraint 

other; in this section, we take weights 

arbitrage 

currency 

exchange rate 


into consideration, as we did for un¬ 
directed edge-weighted graphs in 
section 4.3. Every directed path in 


Typical shortest-paths applications 


edge-weighted digraph 

4- >5 

5- >4 

4- >7 

5- >7 
7->5 

5- >l 
0->4 
0->2 
7->3 

1- >3 

2- >7 

6 - >2 

3- >6 
6->0 
6->4 



0.39 

0.29 

0.34 

0.40 

0.52 

0.58 

93 


shortest path from 0 to 6 
0->2 0.26 

2- >7 0.34 

7->3 0.39 

3- >6 0.52 


0 


an edge-weighted digraph has an associated path weight, 
the value of which is the sum of the weights of that path’s 
edges. This essential measure allows us to formulate such 
problems as “find the lowest-weight directed path from 
one vertex to another,” the topic of this section. The fig¬ 
ure at left shows an example. 

Definition. A shortest path from vertex s to vertex t 
in an edge-weighted digraph is a directed path from 
s to t with the property that no other such path has a 
lower weight. 


An edge-weighted digraph and a shortest path 
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Thus, in this section, we consider classic algorithms for the following problem: 

Single-source shortest paths. Given an edge-weighted digraph and a source ver¬ 
tex s, support queries of the form Is there a directed path from s to a given target 

vertex t? If so, find a shortest such path (one whose total weight is minimal). 

The plan of the section is to cover the following list of topics: 

■ Our APIs and implementations for edge-weighted digraphs, and a single-source 
shortest-paths API 

■ The classic Dijkstra’s algorithm for the problem when weights are nonnegative 

■ A faster algorithm for acyclic edge-weighted digraphs (edge-weighted DAGs) 
that works even when edge weights can be negative 

■ The classic Bellman-Ford algorithm for use in the general case, when cycles may 
be present, edge weights may be negative, and we need algorithms for finding 
negative-weight cycles and shortest paths in edge-weighted digraphs with no 
such cycles 

In the context of the algorithms, we also consider applications. 

Properties of shortest paths The basic definition of the shortest-paths problem 
is succinct, but its brevity masks several points worth examining before we begin to 
formulate algorithms and data structures for solving it: 

■ Paths are directed. A shortest path must respect the direction of its edges. 

■ The weights are not necessarily distances. Geometric intuition can be helpful in 
understanding algorithms, so we use examples where vertices are points in the 
plane and weights are Euclidean distances, such as the digraph on the facing 
page. But the weights might represent time or cost or an entirely different vari¬ 
able and do not need to be proportional to a distance at all. We are emphasizing 
this point by using mixed-metaphor terminology where we refer to a shortest 
path of minimal weight or cost. 

■ Not all vertices need be reachable. If t is not reachable from s, there is no path at 
all, and therefore there is no shortest path from s to t. For simplicity, our small 
running example is strongly connected (every vertex is reachable from every 
other vertex). 

■ Negative weights introduce complications. For the moment, we assume that edge 
weights are positive (or zero). The surprising impact of negative weights is a 
major focus of the last part of this section. 

■ Shortest paths are normally simple. Our algorithms ignore zero-weight edges that 
form cycles, so that the shortest paths they find have no cycles. 

■ Shortest paths are not necessarily unique. There may be multiple paths of the low- 
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0 6->0 

1 null 

2 6->2 

3 l->3 

4 6->4 

5 7->5 

6 3->6 

7 2->7 


0 6->0 

1 5->l 

2 6->2 

3 null 

4 6->4 

5 7->5 

6 3->6 

7 2->7 


0 6->l 

1 5->l 

2 6->2 

3 l->3 

4 5->4 

5 null 

6 3->6 

7 5->7 


0 6->0 

1 5->l 

2 6->2 

3 7->3 

4 5->4 

5 7->5 

6 3->6 

7 null 


parent-edge array 
representation \ 



nul 1 
5->l 
0->2 
7->3 
0->4 
4->5 
3->6 
2->7 


6->0 

5->l 

null 

1- >3 
5->4 
7->5 
3->6 

2- >7 


6->0 

5- >l 

6 - >2 
7->3 
null 
4->5 

3- >6 

4- >7 


6->0 

5- >l 

6 - >2 
7->3 

6- >4 

7- >5 
null 
2->7 


Shortest-paths trees 


est weight from one vertex to another; we are content 
to find any one of them. 

■ Parallel edges and self-loops may be present. Only the 
lowest-weight among a set of parallel edges will play 
a role, and no shortest path contains a self-loop (ex¬ 
cept possibly one of zero weight, which we ignore). 

In the text, we implicitly assume that parallel edges 
are not present for convenience in using the notation 
v->w to refer unambiguously to the edge from v to w, 
but our code handles them without difficulty. 

Shortest-paths tree We focus on the single-source 
shortest-paths problem, where we are given a source ver¬ 
tex s. The result of the computation is a tree known as 
the shortest-paths tree (SPT), which gives a shortest path 
from s to every vertex reachable from s. 

Definition. Given an edge-weighted digraph and 
a designated vertex s, a shortest-paths tree for a 
source s is a subgraph containing s and all the ver¬ 
tices reachable from s that forms a directed tree 
rooted at s such that every tree path is a shortest 
path in the digraph. 


edges point away 
from the source 


Such a tree always exists: in general there may be two 
paths of the same length connecting s to a vertex; if that 
is the case, we can delete the final edge on one of them, 
continuing until we have only 
one path connecting the source 
to each vertex (a rooted tree). 

By building a shortest-paths 
tree, we can provide clients 
with the shortest path from 
s to any vertex in the graph, 
using a parent-link represen¬ 
tation, in precisely the same 
manner as for paths in graphs 
in section 4.1. 



An SPT with 250 vertices 
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Edge-weighted digraph data types Our data type for directed edges is sim¬ 
pler than for undirected edges because we follow directed edges in just one direction. 
Instead of the eitherO and other () methods in Edge, we have from() and to() 
methods: 

public class DirectedEdge 



DirectedEdgefint v, 

int w, double weight) 

double 

weight() 

weight of this edge 

int 

f rom() 

vertex this edge points from 

int 

to() 

vertex this edge points to 

String 

toString() 

string representation 


Weighted directed-edge API 


As with our transition from Graph to EdgeWei ghtedGraph from section 4.1 to sec¬ 
tion 4.3, we include an edges () method and use Di rectedEdge instead of integers: 


public class 

EdgeWeightedDigraph 



EdgeWeightedDigraph(int V) 

empty V-vertex digraph 


EdgeWeightedDigraph(In in) 

construct from i n 

int 

V() 

number of vertices 

int 

E() 

number of edges 

voi d 

addEdge(DirectedEdge e) 

add e to this digraph 

Iterable<DirectedEdge> 

adjfint v) 

edges pointing from v 

Iterable<DirectedEdge> 

edges() 

all edges in this digraph 

String 

toString() 

Edge-weighted digraph API 

string representation 


You can find implementations of these two APIs on the following two pages. These are 
natural extensions of the implementations of section 4.2 and section 4.3. Instead of 
the adjacency lists of integers used in Di graph, we have adjacency lists of Di rectedEdge 
objects in EdgeWei ghtedDi graph. As with the transition from Graph to Di graph from 
section 4.1 to section 4.2, the transition from EdgeWei ghtedGraph in section 4.3 to 
EdgeWei ghtedDi graph in this section simplifies the code, since each edge appears only 
once in the data structure. 
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Directed weighted edge data type 


public class DirectedEdge 

{ 


private final int v; 
private final int w; 
private final double weight; 


// edge source 
// edge target 
// edge weight 


public DirectedEdge(int v, int w, double weight) 

{ 

this.v = v; 
this.w = w; 
this.weight = weight; 

} 

public double weightO 
{ return weight; } 

public int fromO 
{ return v; } 

public int to() 

{ return w; } 

public String toStringO 

{ return String.format("%d->%d %.2f", v, w, weight); } 


} 


This Di rectedEdge implementation is simpler than the undirected weighted Edge implementation 
of section 4.3 (see page 610 ) because the two vertices are distinguished. Our clients use the idiomatic 
codeint v = e.toQ, w = e. from() ; to access a Di rectedEdge e’s two vertices. 
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Edge-weighted digraph data type 


public class EdgeWeightedDigraph 

{ 

private final int V; // number of vertices 

private int E; // number of edges 

private Bag<DirectedEdge>[] adj; // adjacency lists 

public EdgeWeightedDigraph(int V) 

{ 

this.V = V; 
this.E = 0; 

adj = (Bag<DirectedEdge>[]) new Bag[V]; 
for (int v = 0; v < V; v++) 

adj[v] = new Bag<DirectedEdge>(); 

} 

public EdgeWeightedDigraph(In in) 

// See Exercise 4.4.2. 

public int V() { return V; } 
public int E() { return E; } 

public void addEdge(DirectedEdge e) 

{ 

adj[e.from()].add(e); 

E++; 

} 

public Iterable<Edge> adj(int v) 

{ return adj[v]; } 

public Iterable<DirectedEdge> edges() 

{ 

Bag<DirectedEdge> bag = new Bag<DirectedEdge>(); 
for (int v = 0; v < V; v++) 

for (DirectedEdge e : adj[v]) 
bag.add(e); 
return bag; 

} 

} 


This EdgeWei ghtedDi graph implementation is an amalgam of EdgeWei ghtedCraph and Digraph 
that maintains a vertex-indexed array of bags of Di rectedEdge objects. As with Digraph, every 
edge appears just once: if an edge connects v to w, it appears in v’s adjacency list. Self-loops and 
parallel edges are allowed. The toStri ng() implementation is left as exercise 4.4.2. 
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tinyEWD.txt 



45 0.35 

54 0.35 

47 0.37 

5 7 0.28 

7 5 0.28 

5 1 0.32 

0 4 0.38 

0 2 0.26 
7 3 0.39 

1 3 0.29 

2 7 0.34 

62 0.40 

36 0.52 

6 0 0.58 

64 0.93 



Edge-weighted digraph representation 


The figure above shows the data structure that EdgeWei ghtedDi graph builds to rep¬ 
resent the digraph defined by the edges at left when they are added in the order they 
appear. As usual, we use Bag to represent adjacency lists and depict them as linked lists, 
the standard representation. As with the unweighted digraphs of section 4.2, only one 
representation of each edge appears in the data structure. 

Shortest-paths API. For shortest paths, we use the same design paradigm as for the 
DepthFi rstPaths and BreadthFi rstPaths APIs in section 4.1. Our algorithms im¬ 
plement the following API to provide clients with shortest paths and their lengths: 


public class SP 


double 
boolean 

Iterable<DirectedEdge> 


SP(EdgeWeightedDigraph C, 
distTofint v) 
hasPathTofint v) 
pathTofint v) 


i nt s) constructor 

distance from s 
to v, 00 if no path 

path from s to v? 

path from s to v, 
nul 1 if none 


API for shortest-paths implementations 


The constructor builds the shortest-paths tree and computes shortest-paths distances; 
the client query methods use those data structures to provide distances and iterable 
paths to the client. 
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Test client. A sample client is shown below. It takes an input stream and source vertex 
index as command-line arguments, reads the edge-weighted digraph from the input 
stream, computes the SPT of that digraph for the source, and prints the shortest path 
from the source to each of the other 
vertices. We assume that all of our 
shortest-paths implementations in¬ 
clude this test client. Our examples 
use the file tinyEWD.txt shown on 
the facing page, which defines the 
edges and weights that are used in 
the small sample digraph that we 
use for detailed traces of shortest- 
paths algorithms. It uses the same 
file format that we used for MST al¬ 
gorithms: the number of vertices V 
and the number of edges E followed 
by E lines, each with two vertex in¬ 
dices and a weight. You can also find 
on the booksite files that define sev¬ 
eral larger edge-weighted digraphs, 
including the file medi umEWD. txt which defines the 250 -vertex graph drawn on page 640 . 
In the drawing of the graph, every line represents edges in both directions, so this file 
has twice as many lines as the corresponding file medi umEWG. txt that we examined for 
MSTs. In the drawing of the SPT, each line represents a directed edge pointing away 
from the source. 


public static void mai n(String[] args) 

{ 

EdgeWeightedDigraph C; 

C = new EdgeWeightedDigraph(new ln(args[0])); 
int s = Integer.parselnt(args[1]); 

SP sp = new SP(G, s); 

for (int t = 0; t < G.V(); t++) 

{ 

StdOut.print(s + " to " + t); 

StdOut. pri ntf (" (%4.2f): ", sp.distTo(t)) ; 
if (sp.hasPathTo(t)) 

for (DirectedEdge e : sp.pathTo(t)) 
StdOut.print(e + " "); 

StdOut. pri ntln() ; 

} 

} 

Shortest paths test client 


% 

java 

SP tinyEWD.txt 0 





0 

to 

0 

(0.00) 







0 

to 

1 

(1.05) 

0->4 

0.38 

4->5 

0.35 

5->l 

0.32 

0 

to 

2 

(0.26) 

0->2 

0.26 





0 

to 

3 

(0.99) 

0->2 

0.26 

2->7 

0.34 

7->3 

0.39 

0 

to 

4 

(0.38) 

0->4 

0.38 





0 

to 

5 

(0.73) 

0->4 

0.38 

4->5 

0.35 



0 

to 

6 

(1.51) 

0->2 

0.26 

2->7 

0.34 

7->3 

0.39 

0 

to 

7 

(0.60) 

0->2 

0.26 

2->7 

0.34 




3->6 0.52 
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Data structures for shortest paths. The data structures that we need to represent 
shortest paths are straightforward: 

■ Edges on the shortest-paths tree: As for DFS, BFS, and Prim’s algorithm, we use a 
parent-edge representation in the form of a vertex-indexed array edgeTo [] of 
Di rectedEdge objects, where edgeTo [v] is edge that connects v to its parent in 
the tree (the last edge on a shortest path from s to v). 

■ Distance to the source: We use a vertex-indexed array di stTo [] such that 
di stTo [v] is the length of the shortest known path from s to v. 

By convention, edgeTofs] is null and distTofs] is 0. We also adopt the con¬ 
vention that distances to vertices that are not reachable from the source are all 
Doubl e. POSITIVE_INFINITY. As usual, we will develop data types that build these 
data structures in the constructor and 
then support instance methods that use 
them to support client queries for short¬ 
est paths and shortest-path distances. 

Edge relaxation. Our shortest-paths 
implementations are based on a sim¬ 
ple operation known as relaxation. We 
start knowing only the graph’s edges 
and weights, with the distTo[] en¬ 
try for the source initialized to 0 and all of the other distTo[] entries initialized to 
Doubl e. POSITIVE_INFINITY. As an algorithm proceeds, it gathers information about 
the shortest paths that connect the source to each vertex encountered in our edgeTo [] 
and di stTo [] data structures. By updating this information when we encounter edges, 
we can make new inferences about shortest paths. Specifically, we use edge relaxation, 
defined as follows: to relax an edge v->w means to test whether the best known way 
from s to w is to go from s to v, then take the edge from v to w, and, if so, update our 
data structures to indicate that 
to be the case. The code at the 
right implements this opera¬ 
tion. The best known distance 
to w through v is the sum of 
distTo[v] and e.weightO— 
if that value is not smaller than 
distTo[w], we say the edge is 
ineligible, and we ignore it; if it 
is smaller, we update the data 


private void relax(Di rectedEdge e) 

{ 

int v = e.fromO, w = e.to(); 
if (distTo[w] > distTo[v] + e.weightO) 
{ 

distTo[w] = distTo[v] + e.weightO; 
edgeTo[w] = e; 

} 


Edge relaxation 



edgeTo[] 
nul 1 

5->l 0.32 
0->2 
7->3 
0->4 

4->5 0.35 
3->6 0.52 
2->7 


di stTo[] 
0 

1.05 

0.26 

0.97 

0.38 

0.73 

1.49 

0.60 


Shortest-paths data structures 
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structures. The figure at the bottom of this page illustrates the two possible outcomes of 
an edge-relaxation operation. Either the edge is ineligible (as in the example at left) and 
no changes are made, or the edge v->w leads to a shorter path to w (as in the example 
at right) and we update edgeTo[w] and di stTo[w] (which might render some other 
edges ineligible and might create some new eligible edges). The term relaxation follows 
from the idea of a rubber band stretched tight on a path connecting two vertices: relax¬ 
ing an edge is akin to relaxing the tension on the rubber band along a shorter path, if 
possible. We say that an edge e can be successfully relaxed if rel ax() would change the 
values of distTo[e.to()] and edgeTo[e.to()]. 


IZ-VUI ir inalinitila ir olinikln 




Edge relaxation (two cases) 
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Vertex relaxation. All of our implementations 
actually relax all the edges pointing from a given 
vertex as shown in the (overloaded) implementa¬ 
tion of relaxO below. Note that any edge from 
a vertex whose di stTo[v] entry is finite to a ver¬ 
tex whose di stTo [] entry is infinite is eligible and 
will be added to edgeTo[] if relaxed. In particu¬ 
lar, some edge leaving the source is the first to be 
added to edgeTo[]. Our algorithms choose verti¬ 
ces judiciously, so that each vertex relaxation finds 
a shorter path than the best known so far to some 
vertex, incrementally progressing toward the goal 
of finding shortest paths to every vertex. 



O' 


Vertex relaxation 


private void relax(EdgeWeightedDigraph G, int v) 
{ 

for (DirectedEdge e : G.adj(v)) 

{ 

int w = e.to() ; 

if (distTo[w] > distTo[v] + e.weightO) 

{ 

distTo[w] = distTo[v] + e.weightO; 
edgeTo[w] = e; 

} 

} 

} 


Vertex relaxation 
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Client query methods. In a manner similar to our implementations for pathfinding 
APIs in section 4.1 (and exercise 4.1.13), the edgeTo[] and distTo[] data struc¬ 
tures directly support the pathToO, hasPathToO, and di stTo() client query meth¬ 
ods, as shown below. This code is included in all of our shortest-paths implementa¬ 
tions. As we have noted already, distTo[v] is only meaningful when v is reachable 
from s and we adopt the convention that di stTo() should return infinity for vertices 
that are not reachable from s. To implement this convention, we initialize all di stTo [] 
entries to Doubl e. POSITIVE_INFINITY and distTo[s] to 0; then our shortest-paths 
implementations will set di stTo [v] to a finite value for all vertices v that are reachable 
from the source. Thus, we can dispense with the marked [] array that we normally use 
to mark reachable vertices in a graph search and implement hasPathTo(v) by testing 
whether distTo[v] equals Doubl e. POSITIVE„INFINITY. For pathToO, we use the 
convention that pathTo(v) returns null if v is 
not reachable from the source and a path with 
no edges if v is the source. For reachable vertices, 
we travel up the tree, pushing the edges that we 
find on a stack, in the same manner as we did 
for DepthFirstPaths and BreadthFirstPaths. 

The figure at right shows the discovery of the 
path 0 -> 2 -> 7 -> 3->6 for our example. 



edgeTo[] 

null 

5->l 

0->2 

7->3 

0->4 

4->5 

3->6 

2->7 


3->6 


7->3 

3->6 

2->7 

7->3 3->6 

0->2 

2->7 7->3 3->6 

null 

0->2 2->7 7->3 3->6 


Trace of pathToO computation 


public double distTo(int v) 

{ return distTofv]; } 

public boolean hasPathTo(int v) 

{ return distTofv] < Double.POSITIVEJENFINITY; } 

public Iterable<DirectedEdge> pathTo(int v) 

{ 

if (!hasPathTo(v)) return null; 

Stack<DirectedEdge> path = new Stack<DirectedEdge>(); 
for (DirectedEdge e = edgeTofv]; e != null; e = edgeTo[e.from()]) 
path.push(e); 
return path; 

} 

Client query methods for shortest paths 
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Theoretical basis for shortest-paths algorithms. Edge relaxation is an easy- 
to-implement fundamental operation that provides a practical basis for our shortest- 
paths implementations. It also provides a theoretical basis for understanding the algo¬ 
rithms and an opportunity for us to do our algorithm correctness proofs at the outset. 

Optimality conditions. The following proposition shows an equivalence between the 
global condition that the distances are shortest-paths distances, and the local condition 
that we test to relax an edge. 

Proposition P. (Shortest-paths optimality conditions) Let G be an edge-weighted 
digraph, with s a source vertex in G and di stTo [] a vertex-indexed array of path 
lengths in G such that, for all v reachable from s, the value of di stTo[v] is the 
length of some path from s to v with di stTo [v] equal to infinity for all v not reach¬ 
able from s. These values are the lengths of shortest paths if and only if they satisfy 
distTofw] <= distTofv] + e.weightO for each edge e from v to w (or, in 
other words, no edge is eligible). 

Proof: Suppose that distTo[w] is the length of a shortest path from s to w. If 
distTofw] > distTo[v] + e.weightO for some edge e from v to w, then e 
would give a path from s to w (through v) of length less than di stTo [w] , a contra¬ 
diction. Thus the optimality conditions are necessary. 

To prove that the optimality conditions are sufficient, suppose that w is reach¬ 
able from s and that s = v 0 ->v 1 ->v 2 . . . ->v k = w is a shortest path from s to 
w, of weight 0PT sw . For i from 1 to k, denote the edge from to by By the 
optimality conditions, we have the following sequence of inequalities: 

distTofw] = distTo[v k ] <= distTo[v k ^ 1 ] + e k .weight() 

distTo[v k _ 1 ] <= distTo[v k _ 2 ] + e k _ 1 .weight() 

distTo[v 2 ] <= distTofv-J + e 2 .weight() 

distTo[v 1 ] <= distTo[s] + e^^.weightC) 

Collapsing these inequalities and eliminating di stTo [s] = 0.0, we have 

distTo[w] <= e-L.weightO + ... + e k .weight() = 0PT sw . 

Now, di stTo [w] is the length of some path from s to w, so it cannot be smaller than 
the length of a shortest path. Thus, we have shown that 

0PT sw <= di stTo [w] <= 0PT sw 

and equality must hold. 
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Certification. An important practical consequence of proposition p is its applicabil¬ 
ity to certification. However an algorithm computes d i s tTo [ ], we can check whether it 
contains shortest-path lengths in a single pass through the edges of the graph, checking 
whether the optimality conditions are satisfied. Shortest-paths algorithms can be com¬ 
plicated, and this ability to efficiently test their outcome is crucial. We include a method 
check O in our implementations on the booksite for this purpose. This method also 
checks that edgeTo [] specifies paths from the source and is consistent with di stTo []. 

Generic algorithm. The optimality conditions lead immediately to a generic algo¬ 
rithm that encompasses all of the shortest-paths algorithms that we consider. For the 
moment, we restrict attention to nonnegative weights. 

Proposition Q. (Generic shortest-paths algorithm) Initialize distTo[s] to 0 and 
all other di stTo [] values to infinity, and proceed as follows: 

Relax any edge in G, continuing until no edge is eligible. 

For all vertices w reachable from s, the value of di stTo[w] after this computation 
is the length of a shortest path from s to w (and the value of edgeTo [] is the last 
edge on that path). 

Proof: Relaxing an edge v->w always sets distTo[w] to the length of some path 
from s (and edgeTo [w] to the last edge on that path). For any vertex w reachable 
from s, some edge on the shortest path to w is eligible as long as di stTo [w] remains 
infinite, so the algorithm continues until the di stTo [] value of each vertex reach¬ 
able from s is the length of some path to that vertex. For any vertex v for which the 
shortest path is well-defined, throughout the algorithm di stTo [v] is the length of 
some (simple) path from s to v and is strictly monotonically decreasing. Thus, it 
can decrease at most a finite number of times (once for each simple path from s to 
v). When no edge is eligible, proposition p applies. 


The key reason for considering the optimality conditions and the generic algorithm 
is that the generic algorithm does not specify in which order the edges are to be relaxed. 
Thus, all that we need to do to prove that any algorithm computes shortest paths is to 
prove that it relaxes edges until no edge is eligible. 
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Dijkstra’s algorithm In section 4.3, we discussed Prim’s algorithm for finding 
the minimum spanning tree (MST) of an edge-weighted undirected graph: we build 
the MST by attaching a new edge to a single growing tree at each step. Dijkstra’s algo¬ 
rithm is an analogous scheme to compute an SPT. We begin by initializing di st [s] to 
0 and all other di stTo [] entries to positive infinity, then we relax and add to the tree a 
non-tree vertex with the lowest di stTo [] value, continuing until all vertices are on the tree 
or no non-tree vertex has a finite di stTo [] value. 


Proposition R. Dijkstra’s algorithm solves the single-source shortest-paths prob¬ 
lem in edge-weighted digraphs with nonnegative weights. 

Proof: If v is reachable from the source, every edge v->w is relaxed exactly once, 
when v is relaxed, leaving di stTo [w] <= distTo[v] + e .wei ght(). This in¬ 
equality holds until the algorithm completes, since distTo[w] can only decrease 
(any relaxation can only decrease a di stTo [] value) anddistTo[v] never changes 
(because edge weights are nonnegative and we choose the lowest di stTo [] value 
at each step, no subsequent relaxation can set any di stTo [] entry to a lower value 
than di stTo[v]). Thus, after all vertices reachable from s have been added to the 
tree, the shortest-paths optimality conditions hold, and proposition p applies. 


tree edge 


Data structures. To implement Dijkstra’s algorithm we add to our distTo[] and 
edgeTo[] data structures an index priority queue pq to keep track of vertices that are 
candidates for being the next to be relaxed. Recall that an IndexMi nPQ allows us to as¬ 
sociate indices with keys (priorities) and to remove and return the index corresponding 

to the lowest key. For this application, we always associ¬ 
ate a vertex v with di stTo [v] , and we have a direct and 
immediate implementation of Dijkstra’s algorithm as 
stated. Moreover, it is immediate by induction that the 
edgeTo[] entries corresponding to reachable vertices 
form a tree, the SPT. 

Alternative viewpoint. Another way to understand 
the dynamics of the algorithm derives from the proof, 
diagrammed at left: we have the invariant that di stTo [] 
entries for tree vertices are shortest-paths distances and 
for each vertex w on the priority queue, di stTo[w] is 
the weight of a shortest path from s to w that uses only 
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intermediate vertices in the tree and ends in the 
crossing edge edgeTo[w]. The distTo[] entry for 
the vertex with the smallest priority is a shortest- 
path weight, not smaller than the shortest-path 
weight to any vertex already relaxed, and not larger 
than the shortest-path weight to any vertex not yet 
relaxed. That vertex is next to be relaxed. Reachable 
vertices are relaxed in order of the weight of their 
shortest path from s. 

The figure at right is a trace for our small sample 
graph ti nyEWD. txt. For this example, the algo¬ 
rithm builds the SPT as follows: 

■ Adds 0 to the tree and its adjacent vertices 2 
and 4 to the priority queue. 

■ Removes 2 from the priority queue, adds 
0->2 to the tree, and adds 7 to the priority 
queue. 

■ Removes 4 from the priority queue, adds 
0->4 to the tree, and adds 5 to the priority 
queue. Edge 4->7 is ineligible. 

■ Removes 7 from the priority queue, adds 

2- >7 to the tree, and adds B to the priority 
queue. Edge 7->5 is ineligible. 

■ Removes 5 from the priority queue, adds 

4- >5 to the tree, and adds 1 to the priority 
queue. Edge 5->7 is ineligible. 

■ Removes 3 from the priority queue, adds 
7->3 to the tree, and adds 6 to the priority 
queue. 

■ Removes 1 from the priority queue and adds 

5- >l to the tree. Edge l->3 is ineligible. 

■ Removes 6 from the priority queue and adds 

3- >6 to the tree. 

Vertices are added to the SPT in increasing order of 
their distance from the source, as indicated by the 
red arrows at the right edge of the diagram. 
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The implementation of Dijkstra’s algorithm in Di j kstraSP (algorithm 4 . 9 ) is a ren¬ 
dition in code of the one-sentence description of the algorithm, enabled by adding one 
statement to rel ax() to handle two cases: either the to() vertex on an edge is not yet 
on the priority queue, in which case we use i nsertO to add it to the priority queue, 
or it is already on the priority queue and its priority lowered, in which case change () 
does so. 


Proposition R (continued). Dijkstra’s algorithm uses extra space proportional to V 
and time proportional to £ log V (in the worst case) to compute the SPT rooted at 
a given source in an edge-weighted digraph with E edges and V” vertices. 

Proof: Same as for Prim’s algorithm (see proposition n). 


as we have indicated, another way to think about Dijkstra’s algorithm is to 
compare it to Prim’s MST algorithm from section 4.3 (see page 622). Both algorithms 
build a rooted tree by adding an edge to a growing tree: Prim’s adds next the non-tree 
vertex that is closest to the tree; Dijkstra’s adds next the non-tree vertex that is closest 
to the source. The marked [] array is not needed, because the condition ! marked [w] 
is equivalent to the condition that di stTo[w] is infinite. In other words, switching to 
undirected graphs and edges and omitting the references to di stTo [v] in the rel ax O 
code in algorithm 4.9 gives an implementation of algorithm 4 . 7 , the eager version 
of Prim’s algorithm (!). Also, a lazy version of Dijkstra’s algorithm along the lines of 
LazyPrimMST (page 619) is not difficult to develop. 

Variants. Our implementation of Dijkstra’s algorithm, with suitable modifications, is 
effective for solving other versions of the problem, stich as the following: 

Single-source shortest paths in undirected graphs. Given an edge-weighted un¬ 
directed graph and a source vertex s, stipport queries of the form Is there a path 
from s to a given target vertex v? If so, find a shortest such path (one whose total 
weight is minimal). 

The solution to this problem is immediate if we view the undirected graph as a digraph. 
That is, given an undirected graph, build an edge-weighted digraph with the same ver¬ 
tices and with two directed edges (one in each direction) corresponding to each edge 
in the graph. There is a one-to-one correspondence between paths in the digraph and 
paths in the graph, and the costs of the paths are the same—the shortest-paths prob¬ 
lems are equivalent. 
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ALGORITHM 4.9 Dijkstra's shortest-paths algorithm 


public class DijkstraSP 

{ 

private DirectedEdge[] edgeTo; 
private doublet] distTo; 
private IndexMinPQ<Doubl e> pq; 

public DijkstraSP(EdgeWeightedDigraph G, int s) 

{ 

edgeTo = new Di rectedEdge[G.VO] ; 

distTo = new double[G.V()]; 

pq = new IndexMinPQ<Double>(G.V()); 

for (int v = 0; v < G.V(); v++) 

distTo[v] = Double.POSITIVE_INFINITY; 
distTo[s] = 0.0; 

pq.insert(s, 0.0); 
while (!pq.isEmptyO) 
relax(G, pq.delMinO) 

} 

private void relax(EdgeWeightedDigraph G, int v) 

{ 

for(DirectedEdge e : G.adj(v)) 

{ 

int w = e.to(); 

if (distTo[w] > distTo[v] + e.weightO) 

{ 

distTo[w] = distTo[v] + e.weightO; 
edgeTo[w] = e; 

if (pq.contains(w)) pq.change(w, distTo[w]); 
else pq.insert(w, distTo[w]); 

} 

} 

} 

public double distTo(int v) // standard client query methods 

public boolean hasPathTo(int v) // for SPT implementatations 

public Iterable<Edge> pathTo(int v) // (See page 649.) 


This implementation of Dijkstra’s algorithm grows the SPT by adding an edge at a time, always 
choosing the edge from a tree vertex to a non-tree vertex whose destination w is closest to s. 
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Source-sink shortest paths. Given an edge-weighted digraph, a source vertex s, 
and a target vertex t, find the shortest path from s to t. 

To solve this problem, use Dijkstra’s algorithm, but terminate the search as soon as t 
comes off the priority queue. 

All-pairs shortest paths. Given an edge-weighted digraph, support queries of the 
form Given a source vertex s and a target vertex t, is there a path from s to t? If so, 
find a shortest such path (one whose total weight is minimal). 

The surprisingly compact implementation at right below solves the all-pairs shortest 
paths problem, using time and space proportional to E V’log V. It builds an array of 
Di j kstraSP objects, one for each vertex as the source. To answer a client query, it uses 
the source to access the corresponding single-source shortest-paths object and then 
passes the target as argument to the query. 

Shortest paths in Euclidean graphs. Solve the single-source, source-sink, and 
all-pairs shortest-paths problems in graphs where vertices are points in the plane 
and edge weights are proportional to Euclidean distances between vertices. 

A simple modification considerably speeds up Dijkstra’s algorithm in this case (see 
exercise 4 . 4 . 27 ). 

the figures on the facing page show the emergence of the SPT as computed by Di¬ 
jkstra’s algorithm for the Euclidean graph defined by our test file medi umEWD. txt (see 

page 645) for several different sources. Re¬ 
call that line segments in this graph rep¬ 
resent directed edges in both directions. 
Again, these figures illustrate a fascinat¬ 
ing dynamic process. 

Next, we consider shortest-paths algo¬ 
rithms for acyclic edge-weighted graphs, 
where we can solve the problem in linear 
time (faster than Dijkstra’s algorithm) 
and then for edge-weighted digraphs 
with negative weights, where Dijkstra’s 
algorithm does not apply. 


public class Di jkstraAll Pai rsSP 
{ 

private DijkstraSP[] all; 

DijkstraAllPairsSP(EdgeWeightedDigraph G) 
{ 

all = new DijkstraSP[G.V()] 
for (int v = 0; v < G.V(); v++) 
all[v] = new DijkstraSP(G, v); 

} 

Iterable<Edge> path(int s, int t) 

{ return all [s].pathTo(t); } 

double dist(int s, int t) 

{ return al1[s].distTo(t); } 

} 

All-pairs shortest paths 
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Acyclic edge-weighted digraphs For many natural applications, edge-weighted 
digraphs are known to have no directed cycles. For economy, we use the equivalent 
term edge-weighted DAG to refer to an acyclic edge-weighted digraph. We now consider 
an algorithm for finding shortest paths that is simpler and faster than Dijkstra’s algo¬ 
rithm for edge-weighted DAGs. Specifically, it 

■ Solves the single-source problem in linear time 

■ Handles negative edge weights 

■ Solves related problems, such as finding longest paths. 

These algorithms are straightforward extensions to the algorithm for topological sort 


in DAGs that we considered in section 4.2. 

Specifically, vertex relaxation, in combi¬ 
nation with topological sorting, immedi¬ 
ately presents a solution to the single-source 
shortest-paths problem for edge-weighted 
DAGs. We initialize di stTo [s] to 0 and all 
other di stTo [] values to infinity, then relax 
the vertices, one by one, taking the vertices 
in topological order. An argument similar 
to (but simpler than) the argument that we 
used for Dijkstra’s algorithm on page 652 es¬ 
tablishes the effectiveness of this method: 


V- 


tinyEWDAG.txt 



54 0.35 

47 0.37 

57 0.28 

5 1 0.32 

4 0 0.38 

02 0.26 
37 0.39 

13 0.29 

72 0.34 

6 2 0.40 

36 0.52 

6 0 0.58 

64 0.93 



An acyclic edge-weighted digraph with an SPT 


Proposition S. By relaxing vertices in topological order, we can solve the single¬ 
source shortest-paths problem for edge-weighted DAGs in time proportional to 
E+V. 

Proof: Every edge v->w is relaxed exactly once, when v is relaxed, leaving 
distTo[w] <= distTo[v] + e.weight(). This inequality holds until the algo¬ 
rithm completes, since di stTo[v] never changes (because of the topological or¬ 
der, no edge pointing to v will be processed after v is relaxed) and di stTo [w] can 
only decrease (any relaxation can only decrease a di stTo[] value). Thus, after all 
vertices reachable from s have been added to the tree, the shortest-paths optimal¬ 
ity conditions hold, and proposition q applies. The time bound is immediate: 
proposition G on page 583 tells us that the topological sort takes time proportional 
to E+V, and the second relaxation pass completes the job by relaxing each edge 
once, again in time proportional to E+V. 
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The figure at right is a trace for a sample acyclic 
edge-weighted digraph ti nyEWDAG. txt. For this exam¬ 
ple, the algorithm builds the shortest-paths tree from 
vertex 5 as follows: 

■ Does a DFS to discover the topological order 
5 1 B 6 4 7 0 2. 

■ Adds to the tree 5 and all edges leaving it. 

■ Adds to the tree 1 and l->3. 

■ Adds to the tree 3 and 3->6, but not 3->7, which 
is ineligible. 

■ Adds to the tree 6 and edges 6->2 and 6->0, but 
not 6->4, which is ineligible. 

■ Adds to the tree 4 and 4->0, but not 4->7, which 
is ineligible. Edge 6->0 becomes ineligible. 

■ Adds to the tree 7 and 7->2. Edge 6->2 becomes 
ineligible. 

■ Adds 0 to the tree, but not its incident edge 0 -> 2 , 
which is ineligible. 

■ Adds 2 to the tree. 

The addition of 2 to the tree is not depicted; the last 
vertex in a topological sort has no edges leaving it. 

The implementation, shown in algorithm 4 . 10 , 
is a straightforward application of code we have al¬ 
ready considered. It assumes that Topological has 
overloaded methods for the topological sort, using the 
EdgeWei ghtedDi graph and Di rectedEdge APIs of this 
section (see exercise 4 . 4 . 12 ). Note that our boolean 
array marked [] is not needed in this implementation: 
since we are processing vertices in an acyclic digraph 
in topological order, we never re-encounter a vertex 
that we have already relaxed, algorithm 4.10 could 
hardly be more efficient: after the topological sort, the 
constructor scans the graph, relaxing each edge exactly 
once. It is the method of choice for finding shortest 
paths in edge-weighted graphs that are known to be 
acyclic. 

proposition s is significant because it pro¬ 
vides a concrete example where the absence of cycles 
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Trace for shortest paths in an edge-weighted DAG 
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ALGORITHM 4.10 Shortest paths in edge-weighted DAGs 


public class AcyclicSP 

{ 

private DirectedEdge[] edgeTo; 
private doublet] distTo; 

public AcyclicSP(EdgeWeightedDigraph G, int s) 

{ 

edgeTo = new DirectedEdge[G.V()]; 
distTo = new double[G.V()]; 

for (int v = 0; v < G.V(); v++) 

distTo[v] = Double.POSITIVE_INFINITY; 
distTo[s] = 0.0; 

Topological top = new Topological(G); 

for (int v : top.orderO) 
relax(G, v) ; 

} 

private void relax(EdgeWeightedDigraph G, int v) 

// See page 648. 

public double distTo(int v) // standard client query methods 

public boolean hasPathTo(int v) // for SPT implementatations 

public Iterable<Edge> pathTo(int v) // (See page 649.) 

} 


This shortest-paths algorithm for edge-weighted DAGs uses a topological sort (algorithm 4 . 5 , 
adapted to use EdgeWei ghtedDi graph and Di rectedEdge) to enable it to relax the vertices in topo¬ 
logical order, which is all that is needed to compute shortest paths. 
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considerably simplifies a problem. For shortest paths, the topological-sort-based meth¬ 
od is faster than Dijkstra’s algorithm by a factor proportional to the cost of the priority- 
queue operations in Dijkstra’s algorithm. Moreover, the proof of proposition s does 
not depend on the edge weights being nonnegative, so we can remove that restriction 
for edge-weighted DAGs. Next, we consider implications of this ability to allow nega¬ 
tive edge weights, by considering the use of the shortest-paths model to solve two other 
problems, one of which seems at first blush to be quite removed from graph processing. 

Longest paths. Consider the problem of finding the longest path in an edge-weighted 
DAG with edge weights that may be positive or negative. 

Single-source longest paths in edge-weighted DAGs. Given an edge-weighted 
DAG (with negative weights allowed) and a source vertex s, support queries of the 
form: Is there a directed path from s to a given target vertex v? If so, find a longest 
such path (one whose total weight is maximal). 

The algorithm just considered provides a quick solution to this problem: 

Proposition T. We can solve the longest-paths problem in edge-weighted DAGs in 
time proportional to E + V. 

Proof: Given a longest-paths problem, create a copy of the given edge-weighted 
DAG that is identical to the original, except that all edge weights are negated. Then 
the shortest path in this copy is the longest path in the original. To transform the 
solution of the shortest-paths problem to a solution of the longest-paths problem, 
negate the weights in the solution. The running time follows immediately from 
proposition s. 


Using this transformation to develop a class AcycIicLP that finds longest paths in 
edge-weighted DAGs is straightforward. An even simpler way to implement such a 
class is to copy Acycl i cSP, then switch the di stTo [] initialization to Doubl e. NEGA- 
TIVE_INFINITY and switch the sense of the inequality in rel ax(). Either way, we get 
an efficient solution to the longest-paths problem in edge-weighted DAGs. This result is 
to be compared with the fact that the best known algorithm for finding longest simple 
paths in general edge-weighted digraphs (where edge weights may be negative) requires 
exponential time in the worst case (see chapter 6 )! The possibility of cycles seems to 
make the problem exponentially more difficult. 
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The figure at right is a trace of the process 
of finding longest paths in our sample edge- 
weighted DAG ti nyEWDAG. txt, for comparison 
with the shortest-paths trace for the same DAG 
on page 659. For this example, the algorithm 
builds the longest-paths tree (LPT) from vertex 
5 as follows: 

■ Does a DFS to discover the topological 
order 5 1 3 6 4 7 0 2. 

■ Adds to the tree 5 and all edges leaving it. 

■ Adds to the tree 1 and l->3. 

■ Adds to the tree 3 and edges 3->6 and 

3- >7. Edge 5->7 becomes ineligible. 

■ Adds to the tree 6 and edges 6->2, 6->4, 
and 6->0. 

■ Adds to the tree 4 and edges 4->0 and 

4- >7. Edges 6->0 and 3->7 become ineli¬ 
gible. 

■ Adds to the tree 7 and 7->2. Edge 6->2 
becomes ineligible 

■ Adds 0 to the tree, but not 0->2, which is 
ineligible. 

■ Adds 2 to the tree (not depicted). 

The longest-paths algorithm processes the verti¬ 
ces in the same order as the shortest-paths algo¬ 
rithm but produces a completely different result. 


topological sort 

51364702 



edgeTo[] 

0 

1 5->l 

2 

3 

4 5->4 

5 

6 

7 5->7 


0 

1 5->l 

2 

3 l->3 

4 5->4 

5 

6 

7 5->7 


0 

1 5->l 

2 

3 l->3 

4 5->4 

5 

6 3->6 

7 3->7 


0 6->0 

1 5->l 

2 6->2 

3 l->3 

4 6->4 

5 

6 3->6 

7 3->7 


0 4->0 

1 5->l 

2 6->2 

3 l->3 

4 6->4 

5 

6 3->6 

7 4->7 


0 4->0 

1 5->l 

2 7->2 

3 l->3 

4 6->4 

5 

6 3->6 

7 4->7 


0 4->0 

1 5->l 

2 7->2 

3 l->3 

4 6->4 

5 

6 3->6 

7 4->7 


Trace for longest paths in an acyclic network 
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Parallel job scheduling. As an example application, we revisit the class of scheduling 
problems that we first considered in section 4.2 (page 574). Specifically, consider the 
following scheduling problem (differences from the problem on page 575 are italicized): 

Parallel precedence-constrained scheduling. Given a set of jobs of specified du¬ 
ration to be completed, with precedence constraints that specify that certain jobs 
have to be completed before certain other jobs are begun, how can we schedule 
the jobs on identical processors (as many as needed) such that they are all com¬ 
pleted in the minimum amount of time while still respecting the constraints? 

Implicit in the model of section 4.2 is a single processor: we schedule the jobs in to¬ 
pological order and the total time required is the total duration of the jobs. Now, we 
assume that we have sufficient processors to perform as many jobs as possible, limited 
only by precedence constraints. Again, thousands or even millions of jobs might be 
involved, so we require an efficient algorithm. Remarkably, a linear- 
time. algorithm is available—an approach known as the critical path 
method demonstrates that the problem is equivalent to a longest- 
paths problem in an edge-weighted DAG. This method has been used 
successfully in countless industrial applications. 

We focus on the earliest possible time that we can schedule each 
job, assuming that any available processor can handle the job for its 
duration. For example, consider the problem instance specified in 
the table at right. The solution below shows that 173.0 is the mini¬ 
mum possible completion time for any schedule for this problem: the 
schedule satisfies all the constraints, and no schedule can complete 
before time 173.0 because of the job sequence 0->9->6->8->2. This 
sequence is known as a critical path for this problem. Every sequence 
of jobs, each constrained to follow the job just preceding it in the se¬ 
quence, represents a lower bound on the length of the schedule. If we define the length 
of such a sequence to be its earliest possible completion time (total of the durations of 
its jobs), the longest sequence is known as a critical path because any delay in the start¬ 
ing time of any job delays the best achievable completion time of the entire project. 


0 41 70 91 123 173 

Parallel job-scheduling solution 


job duration rnust complete 
before 

0 41.0 179 

1 51.0 2 

2 50.0 

3 36.0 

4 38.0 

5 45.0 

6 21.0 3 8 

7 32.0 3 8 

8 32.0 2 

9 29.0 4 6 

Ajob-scheduling problem 
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Definition. The critical path method for parallel scheduling is to proceed as follows: 
Create an edge-weighted DAG with a source s, a sink t, and two vertices for each 
job (a start vertex and an end vertex). For each job, add an edge from its start vertex 
to its end vertex with weight equal to its duration. For each precedence constraint 
v->w, add a zero-weight edge from the end vertex corresponding tovs to the begin¬ 
ning vertex corresponding to w. Also add zero-weight edges from the source to each 
job’s start vertex and from each job’s end vertex to the sink. Now, schedule each job 
at the time given by the length of its longest path from the source. 


The figure at the top of this page depicts this correspondence for our sample problem, 
and the figure at the bottom of the page gives the longest-paths solution. As specified, 
the graph has three edges for each job (zero-weight edges from the source to the start 
and from the finish to the sink, and an edge from start to finish) and one edge for each 
precedence constraint. The class CPM on the facing page is a straightforward implemen¬ 
tation of the critical path method. It transforms any instance of the job-scheduling 
problem into an instance of the longest-paths problem in an edge-weighted DAG, uses 
Acycl i cLP to solve it, then prints the job start times and schedule finish time. 
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Critical path method for parallel precedence-constrained job scheduling 


public class CPM 

{ 

public static void main(String[] args) 

{ 

int N = Stdln.readlnt(); Stdln.readLine(); 

EdgeWeightedDigraph G; 

G = new EdgeWeightedDigraph(2*N+2); 

int s = 2*N, t = 2*N+1; 
for (int i =0; i < N; i++) 

{ 

Stringf] a = Stdln.readLine().split("\\s+"); 
double duration = Double.parseDouble(a[0]); 
G.addEdge(new DirectedEdge(i, i+N, duration)); 
G.addEdge(new DirectedEdgefs, i, 0.0)); 

G.addEdge(new DirectedEdge(i+N, t, 0.0)); 
for (int j = 1; j < a.length; j++) 

{ 

int successor = Integer.parselnt(a[j]); 

G.addEdge(new DirectedEdge(i+N, successor, 0.0)); 

} 

} 

AcyclicLP Ip = new AcyclicLP(G, s); 

StdOut.println("Start times:"); 
for (int i =0; i < N; i++) 

StdOut.printf("%4d: %5.1f\n", i, lp.distTo(i)); 
StdOut.printf("Finish time: %5.1f\n", 1p.distTo(t)); 

} 

} 


This implementation of the critical path method for job scheduling 
reduces the problem directly to the longest-paths problem in edge- 
weighted DAGs. It builds an edge-weighted digraph (which must be 
a DAG) from the job-scheduling problem specification, as prescribed 
by the critical path method, then uses Acycl i cLP (see proposition t) 
to find the longest-paths tree and to print the longest-paths lengths, 
which are precisely the start times for each job. 




% java CPM < jobsPC.txt 
Start times: 


0 

0 

0 

1 

41 

0 

2 

125 

0 

5 

91 

0 

4 

70 

0 

5 

0 

0 

6 

70 

0 

7 

41 

0 

8 

91 

0 

9 

41 

0 

Finish time: 175.0 


% more jobsPC.txt 
10 

41.0 179 

51.0 2 

50.0 

56.0 

58.0 

45.0 

21.0 5 8 

52.0 58 

52.0 2 

29.0 4 6 
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original 


job 

start 

0 

0.0 

1 

41.0 

2 

123.0 

3 

91.0 

4 

70.0 

5 

0.0 

6 

70.0 

7 

41.0 

8 

91.0 

9 

41.0 

12.0 after 4 

job 

start 

0 

0.0 

1 

41.0 

2 

123.0 

3 

91.0 

4 

111.0 

5 

0.0 

6 

70.0 

7 

41.0 

8 

91.0 

9 

41.0 

' 70.0 after 7 

job 

start 

0 

0.0 

1 

41.0 

2 

123.0 

3 

91.0 

4 

111.0 

5 

0.0 

6 

70.0 

7 

53.0 

8 

91.0 

9 

41.0 


4 by 80.0 after 0 

infeasible! 

Relative deadlines 
in job scheduling 


Proposition U. The critical path method solves the parallel precedence- 
constrained scheduling problem in linear time. 

Proof: Why does the CPM approach work? The correctness of the algo¬ 
rithm rests on two facts. First, every path in the DAG is a sequence of job 
starts and job finishes, separated by zero-weight precedence constraints— 
the length of any path from the source s to any vertex v in the graph is a 
lower bound on the start/fmish time represented by v, because we could 
not do better than scheduling those jobs one after another on the same ma¬ 
chine. In particular, the length of the longest path from s to the sink t is a 
lower bound on the finish time of all the jobs. Second, all the start and finish 
times implied by longest paths ar e feasible —every job starts after the finish 
of all the jobs where it appears as a successor in a precedence constraint, 
because the start time is the length of the longest path from the source to it. 
In particular, the length of the longest path from s to t is an upper bound 
on the finish time of all the jobs. The linear-time performance is immediate 
from proposition t. 


Parallel job scheduling with relative deadlines. Conventional deadlines are 
relative to the start time of the first job. Suppose that we allow an additional 
type of constraint in the job-scheduling problem to specify 
that a job must begin before a specified amount of time has 
elapsed, relative to the start time of another job. Such con¬ 
straints are commonly needed in time-critical manufactur¬ 
ing processes and in many other applications, but they can 
make the job-scheduling problem considerably more diffi¬ 
cult to solve. For example, as shown at left, suppose that we 
need to add a constraint to our example that job 2 must start 
no later than 12 time units after job 4 starts. This deadline is actually a constraint 
on the start time of job 4: it must be no earlier than 12 time units before the start 
time of job 2. In our example, there is room in the schedule to meet the deadline: 
we can move the start time of job 4 to 111, 12 time units before the scheduled 
start time of job 2. Note that, if job 4 were a long job, this change would increase 
the finish time of the whole schedule. Similarly, if we add to the schedule a dead¬ 
line that job 2 must start no later than 70 time units after job 7 starts, there is 
room in the schedule to change the start time of job 7 to 53, without having to 
reschedule jobs 3 and 8. But if we add a deadline that job 4 must start no later 


job 


time 


relative 

to 


2 12.0 4 

2 70.0 7 

4 80.0 0 

Added deadlines 
for job scheduling 
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than 80 time units after job 0, the schedule becomes infeasible: the constraints that 4 
must start no more than 80 time units after job 0 and that job 2 must start no more 
than 12 units after job 4 imply that job 2 must start no more than 93 time units after job 
0, but job 2 must start at least 123 time units after job 0 because of the chain 0 (41 time 
units) precedes 9 (29 time units) precedes 6 (21 time units) precedes 8 (32 time units) 
precedes 2. Adding more deadlines of course multiplies the possibilities and turns an 
easy problem into a difficult one. 

Proposition V. Parallel job scheduling with relative deadlines is a shortest-paths 
problem in edge-weighted digraphs (with cycles and negative weights allowed). 

Proof: Use the same construction as in proposition u, adding an edge for each 
deadline: if job v has to start within d time units of the start of job w, add an edge 
from v to w with negative weight d. Then convert to a shortest-paths problem by 
negating all the weights in the digraph. The proof of correctness applies, provided 
that the schedule is feasible. Determining whether a schedule is feasible is part of the 
computational burden, as you will see. 


This example illustrates that negative weights can play a critical role in practical ap¬ 
plication models. It says that if we can find an efficient solution to the shortest-paths 
problem with negative weights, then we can find an efficient solution to the parallel job 
scheduling problem with relative deadlines. Neither of the algorithms we have consid¬ 
ered can do the job: Dijkstra’s algorithm requires that weights be positive (or zero), and 
algorithm 4 . io requires that the digraph be acyclic. Next, we consider the problem of 
coping with negative edge weights in digraphs that are not necessarily acyclic. 


-70 
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Shortest paths in general edge-weighted digraphs Our job-scheduling- 
with-deadlines example just discussed demonstrates that negative weights are not 
merely a mathematical curiosity; on the contrary, they significantly extend the applica¬ 
bility of the shortest-paths problem as a problem-solving model. Accordingly we now 


consider algorithms for edge-weighted digraphs that may have both cycles and negative 


tinyEWDn.txt 



4- >5 0.35 

5- >4 0.35 

4- >7 0.37 

5- >7 0.28 

7->5 0.28 

5- >l 0.32 

0->4 0.38 

0->2 0.26 
7->3 0.39 

1- >3 0.29 

2- >7 0.34 

6 - >2 - 1.20 

3- >6 0.52 

6->0 -1.40 
6->4 -1.25 



are dashed lines 


shortest-paths tree from 0 



edgeTof] distTo[] 

0 

1 5->l 0.93 

2 0->2 0.26 

3 7->3 0.99 

4 6->4 0.26 

5 4->5 0.61 

6 3->6 1.51 

7 2->7 0.60 


weights. Before doing so, we consider some ba¬ 
sic properties of such digraphs to reset our in¬ 
tuition about shortest paths. The figure at left is 
a small example that illustrates the effects of in¬ 
troducing negative weights on a digraph’s short¬ 
est paths. Perhaps the most important effect is 
that when negative weights are present, low- 
weight shortest paths tend to have more edges 
than higher-weight paths. For positive weights, 
our emphasis was on looking for shortcuts; but 
when negative weights are present, we seek de¬ 
tours that use negative-weight edges. This effect 
turns our intuition in seeking “short” paths into 
a liability in understanding the algorithms, so 
we need to suppress that line of intuition and 
consider the problem on a basic abstract level. 

Strawman I. The first idea that suggests itself is 
to find the smallest (most negative) edge weight, 
then to add the absolute value of that number 


An edge-weighted digraph with negative weights t0 a11 the ed B e Wei B hts to transform the digraph 

into one with no negative weights. This naive 

approach does not work at all, because shortest 
paths in the new digraph bear little relation to shortest paths in the old one. The more 
edges a path has, the more it is penalized by this transformation (see exercise 4 . 4 . 14 ). 


Strawman II. The second idea that suggests itself is to try to adapt Dijkstra’s algorithm 
in some way. The fundamental difficulty with this approach is that the algorithm de¬ 
pends on examining paths in increasing order of their distance from the source. The 
proof in proposition r that the algorithm is correct assumes that adding an edge to a 
path makes that path longer. But any edge with negative weight makes the path shorter, 
so that assumption is unfounded (see exercise 4 . 4 . 14 ). 

Negative cycles. When we consider digraphs that could have negative edge weights, 
the concept of a shortest path is meaningless if there is a cycle in the digraph that 
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tinyEWDnc.txt 


45 0.35 

5 4 -0.66 
47 0.37 

5 7 0.28 

7 5 0.28 

5 1 0.32 

0 4 0.38 

0 2 0.26 
7 3 0.39 

1 3 0.29 

2 7 0.34 

62 0.40 

3 6 0.52 

6 0 0.58 

6 4 0.93 

shortest path from 0 to 6 

0->4->7->5->4->7->5...->l->3->6 

An edge-weighted digraph with a negative cycle 


has negative weight. For example, consid¬ 
er the digraph at left, which is identical to 
our first example except that edge 5->4 has 
weight - . 66 . Then, the weight of the cycle 
4->7->5->4 is 

B7+.28-.66 = -.01 

We can spin around that cycle to generate 
arbitrarily short paths! Note that it is not 
necessary for all the edges on a directed cy¬ 
cle to be of negative weight; what matters is 
the sum of the edge weights. 

Definition. A negative cycle in an edge- 
weighted digraph is a directed cycle whose 
total weight (sum of the weights of its 
edges) is negative. 



Now, suppose that some vertex on a path from s to a 
reachable vertex v is also on a negative cycle. In this case, 
the existence of a shortest path from s to v would be a 
contradiction, because we could use the cycle to construct 
a path with weight lower than any given value. In other 
words, shortest paths can be an ill-posed problem if nega¬ 
tive cycles are present. 

Proposition W. There exists a shortest path from s 
to v in an edge-weighted digraph if and only if there 
exists at least one directed path from s to v and no 
vertex on any directed path from s to v is on a nega¬ 
tive cycle. 

Proof: See discussion above and exercise 4 . 4 . 29 . 


Note that the requirement that shortest paths have no 
vertices on negative cycles implies that shortest paths are 
simple and that we can compute a shortest-paths tree for 
such vertices, as we have done for positive edge weights. 



t 


red outline: no shortest path from s exists 

Shortest-paths possibilities 
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Strawman III. Whether or not there are negative cycles, there exists a shortest simple 
path connecting the source to each vertex reachable from the source. Why not define 
shortest paths so that we seek such paths? Unfortunately, the best known algorithm for 
solving this problem takes exponential time in the worst case (see chapter 6). Gener¬ 
ally, we consider such problems “too difficult to solve” and study simpler versions. 

thus, a well-posed and tractable version of the shortest paths problem in edge- 
weighted digraphs is to require algorithms to 

■ Assign a shortest-path weight of +oo to vertices that are not reachable from the 
source 

■ Assign a shortest-path weight of — oo to vertices that are on a path from the 
source that has a vertex that is on a negative cycle 

■ Compute the shortest-path weight (and tree) for all other vertices 
Throughout this section, we have been placing restrictions on the shortest-paths prob¬ 
lem so that we can develop algorithms to solve it. First, we disallowed negative weights, 
then we disallowed directed cycles. We now adopt these less stringent restrictions and 
focus on the following problems in general digraphs: 

Negative cycle detection. Does a given edge-weighted digraph have a negative 
cycle? If it does, find one such cycle. 

Single-source shortest paths when negative cycles are not reachable. Given an 
edge-weighted digraph and a source s with no negative cycles reachable from s, 
support queries of the form Is there a directed path from s to a given target vertex v? 

If so, find a shortest such path (one whose total weight is minimal). 

to summarize: while shortest paths in digraphs with directed cycles is an ill-posed 
problem and we cannot efficiently solve the problem of finding simple shortest paths 
in such digraphs, we can identify negative cycles in practical situations. For example, 
in a job-scheduling-with-deadlines problem, we might expect negative cycles to be 
relatively rare: constraints and deadlines derive from logical real-world constraints, so 
any negative cycles are likely to stem from an error in the problem statement. Finding 
negative cycles, correcting errors, and then finding the schedule in a problem with no 
negative cycles is a reasonable way to proceed. In other cases, finding a negative cycle is 
the goal of the computation. The following approach, developed by R. Bellman and L. 
Ford in the late 1950s, provides a simple and effective basis for attacking both of these 
problems and is also effective for digraphs with positive weights: 
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Proposition X. (Bellman-Ford algorithm) The following method solves the single¬ 
source shortest-paths problem from a given source s for any edge-weighted di¬ 
graph with V vertices and no negative cycles reachable from s: Initialize di stTo [s] 
to 0 and all other di stTo [] values to infinity. Then, considering the digraph’s edges 
in any order, relax all edges. Make V such passes. 

Proof: For any vertex t that is reachable from s consider a specific shortest path 
from s to t: v 0 ->v 1 ->. . . ->v k , where v 0 is s and v k is t. Since there are no negative 
cycles, such a path exists and k can be no larger than V— 1. We show by induction 
on i that after the i th pass the algorithm computes a shortest path from s to V j. The 
base case (i = 0) is trivial. Assuming the claim to be true for i, v 0 ->v 1 ->. . . ->v i 
is a shortest path from s to v i5 and di stTo [v,] is its length. Now, we relax every 
vertex in the ith pass, including v i; so di stTo[v i+1 ] is no greater than di stTo[v.j] 
plus the weight of v,->v i+1 . Now, after the ith pass, di stTo[v i+1 ] must be equal 
to distToCv^ plus the weight of v n ->v i+1 . It cannot be greater because we relax 
every vertex in the ith pass, in particular v i5 and it cannot be less because that is 
the length of v 0 ->v 1 ->. . . ->v i+1 , a shortest path. Thus the algorithm computes a 
shortest path from s to v i+1 after the (i +l)st pass. 


Proposition W (continued). The Bellman-Ford algorithm takes time proportional 
to EV and extra space proportional to V. 

Proof: Each of the V passes relaxes E edges. 


This method is very general, since it does not specify the order in which the edges are 
relaxed. We now restrict attention to a less general method where we always relax all 
the edges leaving any vertex (in any order). The following code exhibits the simplicity 
of the approach: 

for (int pass = 0; pass < G.V(); pass++) 
for (v = 0; v < G.V(); v++) 

for (DirectedEdge e : G.adj(v)) 
re)ax(e); 

We do not consider this version in detail because it always relaxes VE edges, and a 
simple modification makes the algorithm much more efficient for typical applications. 
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Queue-based Bellman-Ford. Specifically, we 
can easily determine a priori that numerous 
edges are not going to lead to a successful 
relaxation in any given pass: the only edges 
that could lead to a change in distTo[] are 
those leaving a vertex whose distTo[] value 
changed in the previous pass. To keep track of 
such vertices, we use a FIFO queue. The op¬ 
eration of the algorithm for our standard ex¬ 
ample with positive weights is shown at right. 
Shown at the left of the figure are the queue 
entries for each pass (in red), followed by the 
queue entries for the next pass (in black). We 
start with the source on the queue and then 
compute the SPT as follows: 

■ Relax l->3 and put 3 on the queue. 

■ Relax 3->6 and put 6 on the queue. 

■ Relax 6->4, 6->0, and 6->2 and put 4, 

0, and 2 on the queue. 

■ Relax 4->7 and 4->5 and put 7 and 4 on 
the queue. Then relax 0->4 and 0->2, 
which are ineligible. Then relax 2->7 
(and recolor 4->7). 

■ Relax 7->5 (and recolor 4->5) but do 
not put 5 on the queue (it is already 
there). Then relax 7->3, which is ineli¬ 
gible. Then relax 5->l, 5->4, and 5->7, 
which are ineligible, leaving the queue 
empty. 

Implementation. Implementing the Bell- 
man-Ford algorithm along these lines requires 
remarkably little code, as shown in algo¬ 
rithm 4 . 11 . It is based on two additional data 
structures: 

■ A queue q of vertices to be relaxed 

■ A vertex-indexed bool ean array onQ [] 
that indicates which vertices are on the 
queue, to avoid duplicates 


© 

® 


d>— © 

% © 


© 

queue vertices for 
each phase are in red 


3 

6 




edgeTo[] 


l->3 


edgeTo[] 


3->6 


edgeTo[] 

6->0 

6->2 

l->3 

6->4 

3->6 



recolored edge 


edgeTo[] 

0 6->0 
1 

2 6->2 

3 l->3 

4 6->4 

5 4->5 

6 3->6 

7 2->7 



edgeTo[] 

0 6->0 
1 

2 6->2 

3 l->3 

4 6->4 

5 7->5 

6 3->6 

7 2->7 



edgeTo[] 
0 6->0 
1 

2 6->2 

3 l->3 

4 6->4 

5 7->5 

6 3->6 

7 2->7 


Trace of the Bellman-Ford algorithm 
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We start by putting the source s on the queue, then enter a loop where we take a ver¬ 
tex off the queue and relax it. To add vertices to the queue, we augement our rel ax() 
implementation from page 646 to put the vertex pointed to by any edge that successfully 
relaxes onto the queue, as shown in the code at right. The data structures ensure that 

■ Only one copy of each vertex 
appears on the queue 

■ Every vertex whose edgeTo [] 
and di stTo [] values change in 
some pass is processed in the 
next pass 

To complete the implementation, we 
need to ensure that the algorithm ter¬ 
minates after V passes. One way to 
achieve this end is to explicitly keep 
track of the passes. Our implemen¬ 
tation BellmanFordSP (algorithm 
4 . 11 ) uses a different approach that 
we will consider in detail on page 
677: it checks for negative cycles in 
the subset of digraph edges in 
edgeTo [] and terminates if it finds 
one. 


private void relax(EdgeWeightedDigraph G, int v) 

{ 

for (DirectedEdge e : G.adj(v) 

{ 

int w = e.to() ; 

if (distTo[w] > distTo[v] + e.weightO) 

{ 

distTo[w] = distTo[v] + e.weightO; 
edgeTo[w] = e; 

if (!onQ[w]) 

{ 

q.enqueue(w); 
onQ[w] = true; 

} 

} 

if (cost++ % G.V() == 0) 
findNegati veCycleO ; 

} 

} 


Relaxation for Bellman-Ford 


Proposition Y. The queue-based implementation of the Bellman-Ford algorithm 
solves the shortest-paths problem from a given source s (or finds a negative cycle 
reachable from s) for any edge-weighted digraph with V vertices, in time propor¬ 
tional to EV and extra space proportional to V, in the worst case. 

Proof: If there is no negative cycle reachable from s, the algorithm terminates after 
relaxations corresponding to the (V—l)st pass of the generic algorithm described in 
proposition x (since all shortest paths have fewer than V-l edges). If there does 
exist a negative cycle reachable from s, the queue never empties. After relaxations 
corresponding to the Vth pass of the generic algorithm described in proposition 
x the edgeTo [] array has a path with a cycle (connects some vertex w to itself) and 
that cycle must be negative, since the path from s to the second occurrence of w 
must be shorter that the path from s to the first occurrence of w for w to be included 
on the path the second time. In the worst case, the algorithm mimics the general 
algorithm and relaxes all E edges in each of V passes. 
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ALGORITHM 4.11 Bellman-Ford algorithm (queue-based) 


public class BellmanFordSP 
{ 


doublet] distTo; 

// 

length of 

path 

to 

V 

DirectedEdge[] edgeTo; 

// 

last edge 

on path 

to V 

boolean[] onQ; 

// 

Is this vertex 

on 

the queue? 

Queue<Integer> queue; 

// 

vertices 

bei ng 

rel 

1 axed 

int cost; 

// 

number of 

calls 

. to relaxQ 

Iterable<DirectedEdge> cycle 

; // 

negative 

cycl e 

i n 

edgeTo[]? 

Bell manFordSP(EdgeWeightedDig 

raph ( 

1, int s) 





distTo = new double[G.VO]; 
edgeTo = new DirectedEdge[G.V()]; 
onQ = new boolean[G.V()]; 
queue = new Queue<Integer>(); 
for (int v = 0; v < G.V(); v++) 

distTo[v] = Double.POSITIVE_INFINITY; 
distTo[s] = 0.0; 
queue.enqueue(s) ; 
onQ[s] = true; 

while (!queue.isEmptyO && Ithis.hasNegativeCycleO) 

{ 

int v = queue.dequeue(); 
onQ[v] = false; 
relax(v); 

} 


private void relax(int v) 

// See page 673. 


public double distTo(int v) 
public boolean hasPathTo(int v) 
public Iterable<Edge> pathTo(int v) 


// standard client query methods 
// for SPT implementatations 
// (See page 649.) 


private void findNegativeCycleO 
public boolean hasNegativeCycleO 
public Iterable<Edge> negativeCycleO 

// See page 677. 


This implementation of the Bellman-Ford algorithm uses a version of relax() that puts vertices 
pointed to by edges that successfully relax on a FIFO queue (avoiding duplicates) and periodically 
checks for a negative cycle in edgeTo [] (see text). 
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The queue-based Bellman-Ford algorithm is an effective and 
efficient method for solving the shortest-paths problem that is 
widely used in practice, even for the case when edge weights 
are positive. For example, as shown in the diagram at right, our 
250-vertex example is complete in 14 passes and requires fewer 
path-length compares than Dijkstra’s algorithm for the same 
problem. 

Negative weights. The example on the next page traces the 
progress of the Bellman-Ford algorithm in a digraph with nega¬ 
tive weights. We start with the source on q and then compute 
the SPT as follows: 

■ Relax 0->2 and 0->4 and put 2 and 4 on the queue. 

■ Relax 2->7 and put 7 on the queue. Then relax 4->5 and 
put 5 on the queue. Then relax 4->7, which is ineligible. 

■ Relax 7->3 and 5->l and put 3 and 1 on the queue. Then 
relax 5->4 and 5->7, which are ineligible. 

■ Relax 3->6 and put 6 on the queue. Then relax l->3, 
which is ineligible. 

■ Relax 6->4 and put 4 on the queue. This negative-weight 
edge gives a shorter path to 4, so its edges must be relaxed 
again (they were first relaxed in pass 2). The distances 

to 5 and to 1 are no longer valid but will be corrected in 
later passes. 

■ Relax 4->5 and put 5 on the queue. Then relax 4->7, 
which is still ineligible. 

■ Relax 5->l and put 1 on the queue. Then relax 5->4 and 
5->7, which are both still ineligible. 

■ Relax l->3, which is still ineligible, leaving the queue 
empty. 

The shortest-paths tree for this example is a single long path 
from 0 to 1. The edges from 4, 5, and 1 are all relaxed twice for 
this example. Rereading the proof of proposition x in the con¬ 
text of this example is a good way to better understand it. 



10 


13 





Bellman-Ford (250 vertices) 
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tinyEWDn.txt 

4->5 

0.35 

5->4 

0.35 

4->7 

0.37 

5->7 

0.28 

7->5 

0.28 

5->l 

0.32 

0->4 

0.38 

0->2 

0.26 

7->3 

0.39 

l->3 

0.29 

2->7 

0.34 

6->2 

-1.20 

3->6 

0.52 

6->0 

-1.40 

6->4 

-1.25 


queue 

\ 

2 

4 
7 

5 


7 

5 

3 

1 


3 

1 

6 


6 

4 









edgeTo[] 

di stTo[] 

0 



1 



2 

0->2 

0.26 

3 



4 

0->4 

0.38 

5 

4->5 

0.73 

6 



7 

2->7 

0.60 


edgeTo[] 

di stTo[] 

0 



1 

5->l 

1.05 

2 

0->2 

0.26 

3 

7->3 

0.99 

4 

0->4 

0.38 

5 

4->5 

0.73 

6 



7 

2->7 

0.60 


edgeTo[] 

di stTo[] 

0 



1 

5->l 

1.05 

2 

0->2 

0.26 

3 

7->3 

0.99 

4 

0->4 

0.38 

5 

4->5 

0.73 

6 

3->6 

1.51 

7 

2->7 

0.60 


edgeTo[] 

di stTo[] 

0 



1 

5->l 

1.05 

2 

0->2 

0.26 ^ 

3 

7->3 

0.99 

4 

6->4 

0.26 

5 

4->5 

0.73 ^ 

6 

3->6 

1.51 

7 

2->7 

0.60 


edgeTo[] 

di stTo[] 

0 



1 

5->l 

1.05 

2 

0->2 

0.26 

3 

7->3 

0.99 

4 

6->4 

0.26 

5 

4->5 

0.61 

6 

3->6 

1.51 

7 

2->7 

0.60 


edgeTo[] 

di stTo[] 

0 



1 

5->l 

0.93 

2 

0->2 

0.26 

3 

7->3 

0.99 

4 

6->4 

0.26 

5 

4->5 

0.61 

6 

3->6 

1.51 

7 

2->7 

0.60 


no longer eligible ! 


Trace of the Bellman-Ford algorithm (negative weights) 
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Negative cycle detection. Our implementation Bel 1 manFordSP checks for negative 
cycles to avoid an infinite loop. We can apply the code that does this check to provide 
clients with the capability to check for and extract negative cycles, as well. We do so by 
adding the following methods to the SP API on page 644: 


boolean hasNegativeCycleO has a negative cycle? 

Iterabl e<Di rectedEdge> negativeCycleO a negative cycle 

(null if no negative cycles) 

Shortest -paths API extensions for handling negative cycles 


Implementing these methods is not difficult, as shown in the code below. After running 
the constructor in Bel 1 manFordSP, the proof of proposition y tells us that the digraph 
has a negative cycle reachable from the source if and only if the queue is nonempty after 
the Vth pass through all the edges. Moreover, the subgraph of edges in our edgeTo[] 
array must contain a negative cycle. Accordingly, to implement negati veCycl e() we 
build an edge-weighted digraph from the edges in edgeTo[] and look for a cycle in 
that digraph. To find the cycle, we use a version of Di rectedCycl e from section 4 . 3 , 
adapted to work for edge-weighted digraphs (see exercise 4 . 4 . 12 ). We amortize the 
cost of this check by 

■ Adding an instance variable cycl e and a private method f i ndNegati veCycl e () 
that sets cycl e to an iterator for 
the edges of a negative cycle if one 
is found (and to nul 1 if none is 
found) 

■ Calling fi ndNegati veCycl e() 
every Vth call to rel ax() 

This approach ensures that the loop in the 
constructor terminates. Moreover, clients 
can call hasNegativeCycleO to learn 
whether there is a negative cycle reachable 
from the source (and negativeCycleO 
to get one such cycle. Adding the capa¬ 
bility to detect any negative cycle in the 
digraph is also a simple extension (see 
exercise 4 . 4 . 43 ). 

Negative cycle detection methods for Bellman-Ford algorithm 


private void fi ndNegati veCycl e() 

{ 

int V = edgeTo.length; 

EdgeWeightedDigraph spt; 
spt = new EdgeWeightedDigraph(V); 
for (int v = 0; v < V; v++) 
if (edgeTo[v] != null) 
spt.addEdge(edgeTo[v]); 

EdgeWeightedCycleFinder cf; 

cf = new EdgeWeightedCycleFinder(spt); 

cycle = cf .cycleO ; 

} 

public boolean hasNegativeCycleO 
{ return cycle != null; } 

public Iterable<Edge> negativeCycleO 
{ return cycle; } 
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The example below traces the progress of the Bellman-Ford algorithm in a digraph 
with a negative cycle. The first two passes are the same as for tinyEWDn.txt. In the 
third pass, after relaxing 7->3 and 5->l and putting 3 and 1 on queue, it relaxes the 
negative-weight edge 5->4. This relaxation discovers the negative cycle 4->5->4. It puts 
5->4 on the tree and cuts the cycle off from the source 0 in edgeTo []. From that point 
on, the algorithm spins through the cycle, lowering the distances to all the vertices 
touched, until finishing when the cycle is detected, with the queue not empty. The cycle 
is in the edgeTo [] array, for discovery by fi ndNegati veCycleQ. 


tinyEWDnc.txt 

queue 

4->5 

0.35 

\ 

5->4 

-0.66 

2 

4->7 

0.37 

4 

5->7 

0.28 

7 

7->5 

0.28 

5 

5->l 

0.32 


0->4 

0.38 


0->2 

0.26 


7->3 

0.39 


l->3 

0.29 


2->7 

0.34 

7 

6->2 

0.40 

5 

3->6 

0.52 

3 

6->0 

0.58 

1 

6->4 

0.93 

4 


3 

i 

4 
6 
7 

5 


6 

7 

5 

3 
1 

4 


© © 







edgeTo [] 

di stTo[] 

0 



1 



2 

0->2 

0.26 

3 



4 

0->4 

0.38 

5 

4->5 

0.73 

6 



7 

2->7 

0.60 


edgeTo[] 

di stTo[] 

0 



1 

5->l 

1.05 

2 

0->2 

0.26 

3 

7->3 

0.99 

4 

5->4 

0.07 ^ 

5 

4->5 

0.73 

6 



7 

2->7 

0.60 


edgeTo [] 

di stTo[] 

0 



1 

5->l 

1.05 

2 

0->2 

0.26 

3 

7->3 

0.99 

4 

0->4 

0.07 

5 

4->5 

0.42 

6 

3->6 

1.51 

7 

2->7 

0.44 


edgeTo[] 

di stTo[] 

0 



1 

5->l 

0.74 

2 

0->2 

0.26 

3 

7->3 

0.83 

4 

5->4 

-0.59 - 

5 

4->5 

0.73 

6 

3->6 

1.51 

7 

2->7 

0.60 


_ length of 
0->4->5->4 


-——. length of 
0->4->5->4->5->4 


Trace of the Bellman-Ford algorithm (negative cycle) 
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Arbitrage. Consider a market for financial transactions that is based on trading com¬ 
modities. You can find a familiar example in tables that show conversion rates among 
currencies, such as the one in our sample file rates.txt shown here. The first line 
in the file is the number V of currencies; 
then the file has one line per currency, 
giving its name followed by the conver¬ 
sion rates to the other currencies. For 
brevity, this example includes just five 
of the hundreds of currencies that are 
traded on modern markets: U.S. dol¬ 
lars (USD), Euros (EUR), British pounds 
(GBP), Swiss francs (CHF), and Canadian 
dollars (CAD). The tth number on line s represents a conversion rate: the number of 
units of the currency named on row s that is needed to buy 1 unit of the currency 
named on row t. For example, our table says that 1,000 U.S. dollars will buy 741 euros. 
This table is equivalent to a complete edge-weighted digraph with a vertex correspond¬ 
ing to each currency and an edge corresponding to each conversion rate. An edge s->t 
with weight x corresponds to a conversion from s to t at exchange rate x. Paths in the 
digraph specify multistep conversions. For example, com¬ 
bining the conversion just mentioned with an edge t->u 
with weight y gives a path s->t->u that represents a way 
to convert 1 unit of currency s into xy units of curren¬ 
cy u. For example, we might buy 1,012.206 = 741x1.366 
Canadian dollars with our euros. Note that this gives a 
better rate than directly converting from U.S. dollars to 
Canadian dollars. You might expect xy to be equal to the 
weight of s->u in all such cases, but such tables represent 
a complex financial system where such consistency cannot 
be guaranteed. Thus, finding the path from s to u such 
that the product of the weights is maximal is certainly of 
interest. Even more interesting is a case where the product 
of the edge weights is smaller than the weight of the edge 
from the last vertex back to the first. In our example, sup¬ 
pose that the weight of u->s is z and xyz > 1 . Then cycle s->t->u->s gives a way to 
convert 1 unit of currency s into more than 1 unit (xyz) of currency s. In other words, 
we can make a 100(xyz - 1) percent profit by converting from s to t to u back to s. 
For example, if we convert our 1,012.206 Canadian dollars back to US dollars, we get 
1,012.206*.995 = 1,007.14497 dollars, a 7.14497-dollar profit. That might not seem like 


0.741 * 1.366 * .995 = 1.00714497 



% more rates.txt 
5 


USD 

1 

0.741 

0.657 

1.061 

1.005 

EUR 

1.349 

1 

0.888 

1.433 

1.366 

GBP 

1.521 

1.126 

1 

1.614 

1.538 

CHF 

0.942 

0.698 

0.619 

1 

0.953 

CAD 

0.995 

0.732 

0.650 

1.049 

1 
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Arbitrage in currency exchange 


public class Arbitrage 

{ 

public static void main(String[] args) 

{ 

int V = Stdln. readlntO ; 

String[] name = new String[V]; 

EdgeWeightedDigraph C = new EdgeWeightedDigraph(V); 
for (int v = 0; v < V; v++) 

{ 

name[v] = Stdln.readString(); 
for (int w = 0; w < V; w++) 

{ 

double rate = Stdln.readDouble(); 

DirectedEdge e = new DirectedEdge(v, w, -Math.log(rate)); 
C.addEdge(e); 

} 

} 

BellmanFordSP spt = new BellmanFordSP(G, 0); 
if (spt.hasNegativeCycleO) 

{ 

double stake = 1000.0; 

for (Di rectedEdge e : spt.negativeCycleO) 

{ 

StdOut.printf("%10.5f %s ", stake, name[e.from()]); 
stake *= Math.exp(-e.weight()); 

StdOut.printf("= %10.5f %s\n", stake, name[e.to()]); 

} 

} 

else StdOut.println("No arbitrage opportunity"); 

} 

} 


This BellmanFordSP client finds an arbitrage opportunity in a currency exchange table by construct¬ 
ing a complete-graph representation of the exchange table and then using the Bellman-Ford algo¬ 
rithm to find a negative cycle in the graph. 


% java Arbitrage < rates.txt 
1000.00000 USD = 741.00000 EUR 

741.00000 EUR = 1012.20600 CAD 
1012.20600 CAD = 1007.14497 USD 
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much, but a currency trader might have 1 million dollars and be able to execute these 
transactions every minute, which would lead to profits of over $7,000 per minute, or 
$420,000 per hour! This situation is an example of an arbitrage opportunity that would 
allow traders to make unlimited profits were it not for forces outside the model, such 
as transaction fees or limitations on the size of transactions. Even with these forces, 
arbitrage is plenty profitable in the real world. What does this problem have to do with 
shortest paths? The answer to this question is remarkably simple: 

Proposition Z. The arbitrage problem is a negative-cycle-detection problem in 
edge-weighted digraphs. 

Proof: Replace each weight by its logarithm, negated. With this change, comput¬ 
ing path weights by multiplying edge weights in the original problem corresponds 
to adding them in the transformed problem. Specifically, any product w l w 2 .. . w k 
corresponds to a sum — ln(wj) — ln(w 2 ) — ... — ln(w fc ). The transformed edge 
weights might be negative or positive, a path from v to w gives a way of converting 
from currency v to currency w, and any negative cycle is an arbitrage opportunity. 


-ln(.741) —InCl- 366) 


In our example, where all transactions are possible, the digraph is a complete graph, so 
any negative cycle is reachable from any vertex. In general commodity exchanges, some 
edges may be absent, so the one-argument constructor described in exercise 4 . 4.43 is 
needed. No efficient algorithm for finding the best arbi¬ 
trage opportunity (the most negative cycle in a digraph) 
is known (and the graph does not have to be very big 
for this computational burden to be overwhelming), but 
the fastest algorithm to find any arbitrage opportunity 
is crucial—a trader with that algorithm is likely to sys¬ 
tematically wipe out numerous opportunities before the 
second-fastest algorithm finds any. 


\ 

.2998 


\ 

.3119 


-lnC.995) 

I 

.0050 = -.0071 


THE TRANSFORMATION IN THE PROOF of PROPOSITION Z 
is useful even in the absence of arbitrage, because it re¬ 
duces currency conversion to a shortest-paths problem. 
Since the logarithm function is monotonic (and we ne¬ 
gated the logarithms), the product is maximized precisely 
when the sum is minimized. The edge weights might be 
negative or positive, and a shortest path from v to w gives 
a best way of converting from currency v to currency w. 



A negative cycle that represents 
an arbitrage opportunity 
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Perspective The table below summarizes the important characteristics of the 
shortest-paths algorithms that we have considered in this section. The first reason to 
choose among the algorithms has to do with basic properties of the digraph at hand. 
Does it have negative weights? Does it have cycles? Does it have negative cycles? Be¬ 
yond these basic characteristics, the characteristics of edge-weighted digraphs can vary 
widely, so choosing among the algorithms requires some experimentation when more 
than one can apply. 


algorithm 

restriction 

path length compares 
(order of growth) 

extra 

space 

sweet spot 



typical 

worst case 


Dijkstra (eager) 

positive edge 
weights 

Flog V 

Flog V 

V 

worst-case 
guarantee 

topological sort 

edge-weighted 

DAGs 

E+V 

E+V 

V 

optimal for acyclic 

Bellman-Ford 

(queue-based) 

no negative 
cycles 

E+V 

VE 

V 

widely applicable 


Performance characteristics of shortest-paths algorithms 


Historical notes. Shortest-paths problems have been intensively studied and widely 
used since the 1950s. The history of Dijkstra’s algorithm for computing shortest paths 
is similar (and related) to the history of Prim’s algorithm for computing the MST. The 
name Dijkstra’s algorithm is commonly used to refer both to the abstract method of 
building an SPT by adding vertices in order of their distance from the source and to 
its implementation as the optimal algorithm for the adjacency-matrix representation, 
because E. W. Dijkstra presented both in his 1959 paper (and also showed that the same 
approach could compute the MST). Performance improvements for sparse graphs are 
dependent on later improvements in priority-queue implementations that are not spe¬ 
cific to the shortest-paths problem. Improved performance of Dijkstra’s algorithm is 
one of the most important applications of that technology (for example, with a data 
structure known as a Fibonacci heap, the worst-case bound can be reduced to E + Vlog 
V). The Bellman-Ford algorithm has proven to be useful in practice and has found wide 
application, particularly for general edge-weighted digraphs. While the running time of 
the Bellman-Ford algorithm is likely to be linear for typical applications, its worst-case 
running time is VE. The development of a worst-case linear-time shortest-paths algo¬ 
rithm for sparse graphs remains an open problem. The basic Bellman-Ford algorithm 
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was developed in the 1950s by L. Ford and R. Bellman; despite the dramatic strides in 
performance that we have seen for many other graph problems, we have not yet seen 
algorithms with better worst-case performance for digraphs with negative edge weights 
(but no negative cycles). 
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Q&A 


Q. Why define separate data types for undirected graphs, directed graphs, edge-weight¬ 
ed undirected graphs, and edge-weighted digraphs? 

A. We do so both for clarity in client code and for simpler and more efficient imple¬ 
mentation code in unweighted graphs. In applications or systems where all types of 
graphs are to be processed, it is a textbook exercise in software engineering to define an 
ADT from which ADTs can be derived for Graph, the unweighted undirected graphs of 
section 4 . 1 ; Digraph, the unweighted digraphs of section 4 . 2 ; EdgeWei ghtedGraph, 
the edge-weighted undirected graphs of section 4 . 3 ; or EdgeWeightedDigraph, the 
edge-weighted directed graphs of this section. 

Q. How can we find shortest paths in undirected (edge-weighted) graphs? 

A. For positive edge weights, Dijkstra’s algorithm does the job. We just build an 
EdgeWei ghtedDi graph corresponding to the given EdgeWei ghtedGraph (by adding 
two directed edges corresponding to each undirected edge, one in each direction) and 
then run Dijkstra’s algorithm. If edge weights can be negative, efficient algorithms are 
available, but they are more complicated than the Bellman-Ford algorithm. 
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EXERCISES 

4.4.1 True or false: Adding a constant to every edge weight does not change the solu¬ 
tion to the single-source shortest-paths problem. 

4.4.2 Provide an implementation of toStri ng() for EdgeWei ghtedDi graph. 

4.4.3 Develop an implementation of EdgeWei ghtedDi graph for dense graphs that 
uses an adjacency-matrix (two-dimensional array of weights) representation (see ex¬ 
ercise 4 . 3 . 9 ). Ignore parallel edges. 

4.4.4 Draw the SPT for source 0 of the edge-weighted digraph obtained by deleting 
vertex 7 from ti nyEWD.txt (seepage 644), and give the parent-link representation of the 
SPT. Answer the question for the same graph with all edge reversed. 

4.4.5 Change the direction ofedgeO-> 2 inti nyEWD. txt (seepage 644). Drawtwo differ¬ 
ent SPTs that are rooted at 2 for this modified edge-weighted digraph. 

4.4.6 Give a trace that shows the process of computing the SPT of the digraph defined 
in exercise 4 . 4.5 with the eager version of Dijkstra’s algorithm. 

4.4.7 Develop a version of Di jkstraSP that supports a client method that returns a 
second shortest path from s to t in an edge-weighted digraph (and returns null if there 
is only one shortest path from s to t). 

4.4.8 The diameter of a digraph is the length of the maximum-length shortest path 
connecting two vertices. Write a Di jkstraSP client that finds the diameter of a given 
EdgeWei ghtedDi graph that has nonnegative weights. 

4.4.9 The table below, from an old published road map, purports to give the length of 
the shortest routes connecting the cities. It contains an error. Correct the table. Also, add 
a table that shows how to achieve the shortest routes. 



Providence 

Westerly 

New London 

Norwich 

Providence 

- 

53 

54 

48 

Westerly 

53 

- 

18 

101 

New London 

54 

18 

- 

12 

Norwich 

48 

101 

12 

- 
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EXERCISES (continued) 


4.4.10 Consider the edges in the digraph defined in exercise 4 . 4.4 to be undirected 
edges such that each edge corresponds to equal-weight edges in both directions in the 
edge-weighted digraph. Answer exercise 4 . 4.6 for this corresponding edge-weighted 
digraph. 

4.4.1 1 Use the memory-cost model of section 1.4 to determine the amount of mem¬ 
ory used by EdgeWei ghtedDi graph to represent a graph with V vertices and E edges,. 

4.4.12 Adapt the Di rectedCycle and Topological classes from section 4.2 to use 
the Edgewei ghtedDi graph and Di rected Edge APIs of this section, thus implementing 
EdgeWeightedCycleFi nder and EdgeWeightedTopologi cal classes. 

4.4.13 Show, in the style of the trace in the text, the process of computing the SPT 
with Dijkstra’s algorithm for the digraph obtained by removing the edge 5->7 from 
ti nyEWD. txt (see page 644). 

4.4.14 Show the paths that would be discovered by the two strawman approaches de¬ 
scribed on page 668 for the example ti nyEWDn. txt shown on that page. 

4.4.15 What happens to Bellman-Ford if there is a negative cycle on the path from s to 
v and then you call pathTo(v)? 

4.4.16 Suppose that we convert an EdgeWei ghtedGraphintoan EdgeWei ghtedDi graph 
by creating two Di rectedEdge objects in the EdgeWei ghtedDi graph (one in each di¬ 
rection) foreach Edgeinthe EdgeWei ghtedGraph (as described for Dijkstra’s algorithm 
in the Q&A on page 684) and then use the Bellman-Ford algorithm. Explain why this ap¬ 
proach fails spectacularly. 

4.4.17 What happens if you allow a vertex to be enqueued more than once in the same 
pass in the Bellman-Ford algorithm? 

Answer : The running time of the algorithm can go exponential. For example, describe 
what happens for the complete edge-weighted digraph whose edge weights are all - 1 . 

4.4.18 Write a CPM client that prints all critical paths. 

4.4.19 Find the lowest-weight cycle (best arbitrage opportunity) in the example shown 
in the text. 
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4.4.20 Find a currency-conversion table online or in a newspaper. Use it to build an 
arbitrage table. Note: Avoid tables that are derived (calculated) from a few values and 
that therefore do not give sufficiently accurate conversion information to be interesting. 
Extra credit: Make a killing in the money-exchange market! 

4.4.21 Show, in the style of the trace in the text, the process of computing the SPT with 
the Bellman-Ford algorithm for the edge-weighted digraph of exercise 4 . 4 . 5 . 
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| CREATIVE PROBLEMS 


4.4.22 Vertex weights. Show that shortest-paths computations in edge-weighted di¬ 
graphs with nonnegative weights on vertices (where the weight of a path is defined to 
be the sum of the weights of the vertices) can be handled by building an edge-weighted 
digraph that has weights on only the edges. 

4.4.23 Source-sink shortest paths. Develop an API and implementation that use a ver¬ 
sion of Dijkstra’s algorithm to solve the source-sink shortest path problem on edge- 
weighted digraphs. 

4.4.24 Multisource shortest paths. Develop an API and implementation that uses Di¬ 
jkstra’s algorithm to solve the multisource shortest-paths problem on edge-weighted 
digraphs with positive edge weights: given a set of sources, find a shortest-paths forest 
that enables implementation of a method that returns to clients the shortest path from 
any source to each vertex. Hint: Add a dummy vertex with a zero-weight edge to each 
source, or initialize the priority queue with all sources, with their di stTo [] entries set 
to 0. 


4.4.25 Shortest path between two subsets. Given a digraph with positive edge weights, 
and two distinguished subsets of vertices S and T, find a shortest path from any vertex 
in S to any vertex in T. Your algorithm should run in time proportional to E log V, in 
the worst case. 

4.4.26 Single-source shortest paths in dense graphs. Develop a version of Dijkstra’s al¬ 
gorithm that can find the SPT from a given vertex in a dense edge-weighted digraph in 
time proportional to V 2 . Use an adjacency-matrix representation (see exercise 4.4.3 
and exercise 4.3.29). 

4.4.27 Shortest paths in Euclidean graphs. Adapt our APIs to speed up Dijkstra’s algo¬ 
rithm in the case where it is known that vertices are points in the plane. 

4.4.28 Longest paths in DAGs. Develop an implementation Acycl i cLP that can solve 
the longest-paths problem in edge-weighted DAGs, as described in proposition t. 

4.4.29 General optimality. Complete the proof of proposition w by showing that if 
there exists a directed path from s to v and no vertex on any path from s to v is on a 
negative cycle, then there exists a shortest path from s to v (Hint: See proposition p.) 

4.4.30 All-pairs shortest path in graphs with negative cycles. Articulate an API like the 
one implemented on page 656 for the all-pairs shortest-paths problem in graphs with no 
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negative cycles. Develop an implementation that runs a version of Bellman-Ford to 
identify weights pi [v] such that for any edge v->w, the edge weight plus the differ¬ 
ence between pi [v] and pi [w] is nonnegative. Then use these weights to reweight 
the graph, so that Dijkstra’s algorithm is effective for finding all shortest paths in the 
reweighted graph. 

4.4.31 All-pairs shortest path on a line. Given a weighted line graph (undirected con¬ 
nected graph, all vertices of degree 2, except two endpoints which have degree 1), devise 
an algorithm that preprocesses the graph in linear time and can return the distance of 
the shortest path between any two vertices in constant time. 

4.4.32 Parent-checking heuristic. Modify Bellman-Ford to visit a vertex v only if its 
SPT parent edgeTo[v] is not currently on the queue. This heuristic has been reported 
by Cherkassky, Goldberg, and Radzik to be useful in practice. Prove that it correctly 
computes shortest paths and that the worst-case running time is proportional to EV. 

4.4.33 Shortest path in a grid. Given an N-by-N matrix of positive integers, find the 
shortest path from the ( 0 , 0) entry to the (N— 1 , N— 1 ) entry, where the length of the 
path is the sum of the integers in the path. Repeat the problem but assume you can only 
move right and down. 

4.4.34 Monotonic shortest path. Given a weighted digraph, find a monotonic shortest 
path from s to every other vertex. A path is monotonic if the weight of every edge on 
the path is either strictly increasing or strictly decreasing. The path should be simple 
(no repeated vertices). Hint: Relax edges in ascending order and find a best path; then 
relax edges in descending order and find a best path. 

4.4.35 Bitonic shortest path. Given a digraph, find a bitonic shortest path from s to ev¬ 
ery other vertex (if one exists). A path is bitonic if there is an intermediate vertex v such 
that the edges on the path from s to v are strictly increasing and the edges on the path 
from v to t are strictly decreasing. The path should be simple (no repeated vertices). 

4.4.36 Neighbors. Develop an SP client that finds all vertices within a given distance 
d of a given vertex in a given edge-weighted digraph.The running time of your method 
should be proportional to the size of the subgraph induced by those vertices and the 
vertices incident on them, or V” (to initialize data structures), whichever is larger. 
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4.4.37 Critical edges. Develop an algorithm for finding an edge whose removal causes 
maximal increase in the shortest-paths length from one given vertex to another given 
vertex in a given edge-weighted digraph. 

4.4.38 Sensitivity. Develop an SP client that performs a sensitivity analysis on the 
edge-weighted digraph’s edges with respect to a given pair of vertices s and t: Compute 
a V-by-D boolean matrix such that, for every v and w, the entry in row v and column w 
is true if v->w is an edge in the edge-weighted digraphs whose weight can be increased 
without the shortest-path length from v to w being increased and is fal se otherwise. 

4.4.39 Lazy implementation ofDijkstra’s algorithm. Develop an implementation of the 
lazy version of Dijkstra’s algorithm that is described in the text. 

4.4.40 Bottleneck SPT. Show that an MST of an undirected graph is equivalent to a 
bottleneck SPT of the graph: For every pair of vertices v and w, it gives the path connect¬ 
ing them whose longest edge is as short as possible. 

4.4.41 Bidirectional search. Develop a class for the source-sink shortest-paths prob¬ 
lem that is based on code like algorithm 4.9 but that initializes the priority queue with 
both the source and the sink. Doing so leads to the growth of an SPT from each vertex; 
your main task is to decide precisely what to do when the two SPTs collide. 

4.4.42 Worst case (Dijkstra). Describe a family of graphs with V vertices and E edges 
for which the worst-case running time ofDijkstra’s algorithm is achieved. 

4.4.43 Negative cycle detection. Suppose that we add a constructor to algorithm 4.11 
that differs from the constructor given only in that it omits the second argument and 
that it initializes all di stTo [] entries to 0. Show that, if a client uses that constructor, a 
client call to hasNegati veCycleO returns true if and only if the graph has a negative 
cycle (and negativeCycleO returns that cycle). 

Answer: Consider a digraph formed from the original by adding a new source with an 
edge of weight 0 to all the other vertices. After one pass, all di stTo [] entries are 0, and 
finding a negative cycle reachable from that source is the same as finding a negative 
cycle anywhere in the original graph. 

4.4.44 Worst case (Bellman-Ford). Describe a family of graphs for which algorithm 
4.11 takes time proportional to VE. 
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4.4.45 Fast Bellman-Ford. Develop an algorithm that breaks the linearithmic running 
time barrier for the single-source shortest-paths problem in general edge-weighted di¬ 
graphs for the special case where the weights are integers known to be bounded in 
absolute value by a constant. 

4.4.46 Animate. Write a client program that does dynamic graphical animations of 
Dijkstra’s algorithm. 
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4.4.47 Random sparse edge-weighted digraphs. Modify your solution to exercise 

4.3.34 to assign a random direction to each edge. 

4.4.48 Random Euclidean edge-weighted digraphs. Modify your solution to exercise 

4.3.35 to assign a random direction to each edge. 

4.4.49 Random grid edge-weighted digraphs. Modify your solution to exercise 4.3.36 
to assign a random direction to each edge. 

4.4.50 Negative weights I. Modify your random edge-weighted digraph generators 
to generate weights between x and y (where x and y are both between —1 and 1) by 
rescaling. 

4.4.51 Negative weights II. Modify your random edge-weighted digraph generators to 
generate negative weights by negating a fixed percentage (whose value is supplied by the 
client) of the edge weights. 

4.4.52 Negative weights III. Develop client programs that use your edge-weighted 
digraph to produce edge-weighted digraphs that have a large percentage of negative 
weights but have at most a few negative cycles, for as large a range of values of V and E 
as possible. 
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Testing all algorithms and studying all parameters against all edge-weighted digraph mod¬ 
els is unrealistic. For each problem listed below, write a client that addresses the problem 
for any given input digraph, then choose among the generators above to run experiments 
for that graph model. Use your judgment in selecting experiments, perhaps in response to 
results of previous experiments. Write a narrative explaining your results and any conclu¬ 
sions that might be drawn. 

4 . 4.53 Prediction. Estimate, to within a factor of 10 , the largest graph with E = 10 V 7 
that your computer and programming system could handle if you were to use Dijkstra’s 
algorithm to compute all its shortest paths in 10 seconds. 

4 . 4.54 Cost of laziness. Run empirical studies to compare the performance of the lazy 
version of Dijkstra’s algorithm with the eager version, for various edge-weighted di¬ 
graph models. 

4 . 4.55 Johnsons algorithm. Develop a priority-queue implementation that uses a d- 
way heap. Find the best value of d for various edge-weighted digraph models. 

4 . 4.56 Arbitrage model. Develop a model for generating random arbitrage problems. 
Your goal is to generate tables that are as similar as possible to the tables that you used 
in exercise 4.4.20. 

4 . 4.57 Parallel job-scheduling-with-deadlines model. Develop a model for generating 
random instances of the parallel job-scheduling-with-deadlines problem. Your goal is 
to generate nontrivial problems that are likely to be feasible. 
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W e communicate by exchanging strings of characters. Accordingly, numerous 
important and familiar applications are based on processing strings. In this 
chapter, we consider classic algorithms for addressing the underlying com¬ 
putational challenges surrounding applications such as the following: 

Information processing. When you search for web pages containing a given keyword, 
you are using a string-processing application. In the modern world, virtually all in¬ 
formation is encoded as a sequence of strings, and the applications that process it are 
string-processing applications of crucial importance. 

Genomics. Computational biologists work with a genetic code that reduces DNA to 
(very long) strings formed from four characters (A, C, T, and C). Vast databases giving 
codes describing all manner of living organisms have been developed in recent years, 
so that string processing is a cornerstone of modern research in computational biology. 
Communications systems. When you send a text message or an email or download 
an ebook, you are transmitting a string from one place to another. Applications that 
process strings for this purpose were an original motivation for the development of 
string-processing algorithms. 

Programming systems. Programs are strings. Compilers, interpreters, and other appli¬ 
cations that convert programs into machine instructions are critical applications that 
use sophisticated string-processing techniques. Indeed, all written languages are ex¬ 
pressed as strings, and another motivation for the development of string-processing 
algorithms was the theory of formal languages, the study of describing sets of strings. 

This list of a few significant examples illustrates the diversity and importance of string¬ 
processing algorithms. 
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The plan of this chapter is as follows: After addressing basic properties of strings, we 
revisit in sections 5.1 and 5.2 the sorting and searching APIs from chapters 2 and 3. 
Algorithms that exploit special properties of string keys are faster and more flexible 
than the algorithms that we considered earlier. In section 5.3 we consider algorithms 
for substring search, including a famous algorithm due to Knuth, Morris, and Pratt. 
In section 5.4 we introduce regular expressions, the basis of the pattern-matching 
problem, a generalization of substring search, and a quintessential search tool known 
as grep. These classic algorithms are based on the related conceptual devices known as 
formal languages and finite automata, section 5.5 is devoted to a central application: 
data compression, where we try to reduce the size of a string as much as possible. 

Rules of the game For clarity and efficiency, our implementations are expressed 
in terms of the Java Stri ng class, but we intentionally use as few operations as possible 
from that class to make it easier to adapt our algorithms for use on other string-like 
types of data and to other programming languages. We introduced strings in detail in 
section 1.2 but briefly review here their most important characteristics. 

Characters. A Stri ng is a sequence of characters. Characters are of type char and can 
have one of 2 16 possible values. For many decades, programmers restricted attention to 
characters encoded in 7-bit ASCII (see page 815 for a conversion table) or 8-bit extended 
ASCII, but many modern applications call for 16-bit Unicode. 

Immutability. Stri ng objects are immutable, so that we can use them in assignment 
statements and as arguments and return values from methods without having to worry 
about their values changing. 

Indexing. The operation that we perform most often is extract a specified character 
from a string that the charAtO method in Java’s String class provides. We expect 
charAt () to complete its work in constant time, as if the string were stored in a char [] 
array. As discussed in chapter 1, this expectation is quite reasonable. 

Length. In Java, the find the length of a string operation is implemented in the 1 ength 0 
method in Stri ng. Again, we expect 1 ength () to complete its work in constant time, 
and again, this expectation is reasonable, although some care is needed in some pro¬ 
gramming environments. 

Substring. Java’s substri ng() method implements the extract a specified substring op¬ 
eration. Again, we expect a constant- time implementation of this method, as in Java’s 
standard implementation. If you are not familiar with substri ng () and the reason that 
it is constant-time, be sure to reread our discussion of Java’s standard string implementa¬ 
tion in section 1.2 (see page 80 and page 204). 
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Concatenation. In Java, the create a new 
string formed by appending one string to 
another operation is a built-in operation 
(using the + operator) that takes time pro¬ 
portional to the length of the result. For 
example, we avoid forming a string by ap¬ 
pending one character at a time because 
that is a quadratic process in Java. (Java has 
a Stri ngBui 1 der class for that use.) 


s. lengthO 

I 

0123456789 10 11 12 
s —► ATTACKATDAWN 

t \ 

s.charAt(3) \ 

s.substrings, 11) 

Fundamental constant-time String operations 


Character arrays. The Java String is decidedly not a primitive type. The standard 
implementation provides the operations just described to facilitate client program¬ 
ming. By contrast, many of the algorithms that we consider can work with a low-level 
representation such as an array of char values, and many clients might prefer such a 
representation, because it consumes less space and takes less time. For several of the 
algorithms that we consider, the cost of converting from one representation to the other 
would be higher than the cost of running the algorithm. As indicated in the table below, 
the differences in code that processes the two representations are minor (substri ng() 
is more complicated and is omitted), so use of one representation or the other is no 
barrier to understanding the algorithm. 


understanding the efficiency of these operations is a key ingredient in under¬ 
standing the efficiency of several string-processing algorithms. Not all programming 
languages provide String implementations with these performance characteristics. 
For example, the substring operation and determining the length of a string take time 
proportional to the number of characters in the string in the widely used C program¬ 
ming language. Adapting the algorithms that we describe to such languages is always 
possible (implement an ADT like Java’s Stri ng), but also might present different chal¬ 
lenges and opportunities. 


operation 

array of characters 

Java string 

declare 

char[] a 

String s 

indexed character access 

a [ i ] 

s.charAt(i) 

length 

a. 1ength 

s.lengthO 

convert 

a = s. toCharArrayO ; 

Two ways to represent strings in Java 

s = new Stringf 
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We primarily use the Stri ng data type in the text, with liberal use of indexing and 
length and occasional use of substring extraction and concatenation. When appropri¬ 
ate, we also provide on the booksite the corresponding code for char arrays. In perfor¬ 
mance-critical applications, the primary consideration in choosing between the two for 
clients is often the cost of accessing a character (a[i ] is likely to be much faster than 
s. charAtfi ) in typical Java implementations). 

Alphabets Some applications involve strings taken from a restricted alphabet. In 
such applications, it often makes sense to use an A1 phabet class with the following API: 


public class Alphabet 



A1phabet(String s) 

create a new alphabet from chars in s 

char 

toCharfint index) 

convert index to corresponding alphabet char 

int 

tolndexfchar c) 

convert c to an index between 0 and R-l 

boolean 

containsfchar c) 

is c in the alphabet? 

int 

RO 

radix (number of characters in alphabet) 

int 

lgRO 

number of bits to represent an index 

int[] 

toIndices(String s) 

convert s to base-R integer 

String 

toChars(int[] indices) 

convert base-R integer to string over this alphabet 


Alphabet API 


This API is based on a constructor that takes as argument an ft-character string that 
specifies the alphabet and the toCharO and toIndexO methods for converting (in 
constant time) between string characters and int values between 0 and ft-1. It also 
includes a contai ns () method for checking whether a given character is in the alpha¬ 
bet, the methods R() and lgR() for finding the number of characters in the alphabet 
and the number of bits needed to represent them, and the methods tolndi ces() and 
toChars () for converting between strings of characters in the alphabet and i nt arrays. 
For convenience, we also include the built-in alphabets in the table at the top of the 
next page, which you can access with code such as A1 phabet. UNICODE. Implementing 
A1 phabet is a straightforward exercise (see exercise 5.1.12). We will examine a sample 
client on page 699. 

Character-indexed arrays. One of the most important reasons to use A1 phabet is that 
many algorithms gain efficiency through the use of character-indexed arrays, where 
we associate information with each character that we can retrieve with a single array 
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name 

RO 

lgRO 

characters 

BINARY 

2 

1 

01 

DNA 

4 

2 

ACTG 

OCTAL 

8 

3 

01234567 

DECIMAL 

10 

4 

0123456789 

HEXADECIMAL 

16 

4 

0123456789ABCDEF 

PROTEIN 

20 

5 

ACDEFGHIKLMNPQRSTVWY 

LOWERCASE 

26 

5 

abcdefghij klmnopqrstuvwxyz 

UPPERCASE 

26 

5 

ABCDEFGHI3KLMN0PQRSTUVWXYZ 

BASE64 

64 

6 

ABCDEFGHI!KLMNOPQRSTUVWXYZabcdef 
ghijklmnopqrstuvwxyz0123456789+/ 

ASCII 

128 

7 

ASCII characters 

EXTENDED_ASCII 

256 

8 

extended ASCII characters 

UNIC0DE16 

65536 

16 

Unicode characters 


Standard alphabets 


public class Count 

{ 

public static void main(String[] args) 

{ 

Alphabet alpha = new A1phabet(args[0]); 

int R = al pha. R() ; 

int[] count = new int[R]; 

String s = Stdln. readAl 1 () ; 

int N = s.1ength(); 

for (int i =0; i < N; i++) 

if (alpha.contains(s.charAt(i))) 

count[alpha.toIndex(s.charAt(i))]++; 

for (int c = 0; c < R; C++) 

StdOut. pri ntl n (al pha. toChar(c) 

+ " " + count[c]); 


} 

} 

Typical Alphabet client 


% more abra.txt 
ABRACADABRA! 

% java Count ABCDR < abra.txt 
A 5 
B 2 
C 1 
D 1 
R 2 
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access. With a Java Stri ng, we have to use an array of size 65,536; with A1 phabet, we 
just need an array with one entry for each alphabet character. Some of the algorithms 
that we consider can produce huge numbers of such arrays, and in such cases, the space 
for arrays of size 65,536 can be prohibitive. As an example, consider the class Count at 
the bottom of the previous page, which takes a string of characters from the command 
line and prints a table of the frequency of occurrence of those characters that appear on 
standard input. The count [] array that holds the frequencies in Count is an example of 
a character-indexed array. This calculation may seem to you to be a bit frivolous; actu¬ 
ally, it is the basis for a family of fast sorting methods that we will consider in section 
5-1- 

Numbers. As you can see from several of the standard A1 phabet examples, we often 
represent numbers as strings. The method toIndicesO converts any String over a 
given Alphabet into a base-ft number represented as an int[] array with all values 
between 0 and ft — 1. In some situations, doing this conversion at the start leads to com¬ 
pact code, because any digit can be used as an index in a character-indexed array. For 
example, if we know that the input consists only of characters from the alphabet, we 
could replace the inner loop in Count with the more compact code 

int[] a = alpha.tolndices(s); 

for (int i =0; i < N; i++) 
count[a[i]]++; 

In this context, we refer to ft as the radix, the base of the number system. Several of the 
algorithms that we consider are often referred to as “radix” methods because they work 
with one digit at a time. 


% more pi.txt 

3141592653 

5897932384 

6264338327 

9502884197 

... [100,000 digits of pi] 

% java Count 0123456789 < pi.txt 
0 9999 

1 10137 

2 9908 

3 10026 

4 9971 

5 10026 

6 10028 

7 10025 

8 9978 

9 9902 
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despite the advantages of using a data type such as A1 phabet in string-processing 
algorithms (particularly for small alphabets), we do not develop our implementations 
in the book for strings taken from a general A1 phabet because 

■ The preponderance of clients just use Stri ng 

■ Conversion to and from indices tends to fall in the inner loop and slow down 
implementations considerably 

■ The code is more complicated, and therefore more difficult to understand 
Accordingly we use Stri ng, use the constant R = 256 in the code and R as a parameter 
in the analysis, and discuss performance for general alphabets when appropriate. You 
can find full A1 phabet-based implementations on the booksite. 
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for many sorting applications, the keys that define the order are strings. In this 
section, we look at methods that take advantage of special properties of strings to de¬ 
velop sorts for string keys that are more efficient than the general-purpose sorts that we 
considered in chapter 2 . 

We consider two fundamentally different approaches to string sorting. Both of them 
are venerable methods that have served programmers well for many decades. 

The first approach examines the characters in the keys in a right-to-left order. Such 
methods are generally referred to as least-significant-digit (LSD) string sorts. Use of 
the term digit instead of character traces back to the application of the same basic meth¬ 
od to numbers of various types. Thinking of a string as a base-256 number, considering 
characters from right to left amounts to considering first the least significant digits. 
This approach is the method of choice for string-sorting applications where all the keys 
are the same length. 

The second approach examines the characters in the keys in a left-to-right order, 
working with the most significant character first. These methods are generally referred 
to as most-significant-digit (MSD) string sorts—we will consider two such methods 
in this section. MSD string sorts are attractive because they can get a sorting job done 
without necessarily examining all of the input characters. MSD string sorts are similar 
to quicksort, because they partition the array to be sorted into independent pieces such 
that the sort is completed by recursively applying the same method to the subarrays. 
The difference is that MSD string sorts use just the first character of the sort key to 
do the partitioning, while quicksort uses comparisons that could involve examining 
the whole key. The first method that we consider creates a partition for each character 
value; the second always creates three partitions, for sort keys whose first character is 
less than, equal to, or greater than the partitioning key’s first character. 

The number of characters in the alphabet is an important parameter when analyz¬ 
ing string sorts. Though we focus on extended ASCII strings (R = 256), we will also 
consider strings taken from much smaller alphabets (such as genomic sequences) and 
from much larger alphabets (such as the 65,536-character Unicode alphabet that is an 
international standard for encoding natural languages). 
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Key-indexed counting As a warmup, we consider a sim¬ 
ple method for sorting that is effective whenever the keys are 
small integers. This method, known as key-indexed counting, is 
useful in its own right and is also the basis for two of the three 
string sorts that we consider in this section. 

Consider the following data-processing problem, which 
might be faced by a teacher maintaining grades for a class with 
students assigned to sections, which are numbered 1, 2, B, and 
so forth. On some occasions, it is necessary to have the class 
listed by section. Since the section numbers are small integers, 
sorting by key-indexed counting is appropriate. To describe 
the method, we assume that the information is kept in an array 
a [] of items that each contain a name and a section number, 
that section numbers are integers between 0 and R-l, and that 

the code a[i] .keyO returns 
the section number for the in¬ 
dicated student. The method 
breaks down into four steps, 
which we describe in turn. 


for (i =0; i < N; 
count [a[i ] . key() 


i++) 

+ 1]H 


always 0 - 


count [] 
1 2 B 4 


5.1 


String Sorts 

input 


sorted resull 

name section (by section 

Anderson 

2 

Harri s 

Brown 

B 

Marti n 

Davi s 

3 

Moore 

Garci a 

4 

Anderson 

Harri s 

1 

Martinez 

Jackson 

3 

Miller 

Johnson 

4 

Robinson 

Jones 

3 

Whi te 

Marti n 

1 

Brown 

Martinez 

2 

Davi s 

Miller 

2 

Jackson 

Moore 

1 

Jones 

Robinson 

2 

Tayl or 

Smith 

4 

Williams 

Taylor 

3 

Garci a 

Thomas 

4 

Johnson 

Thompson 

4 

Smith 

Whi te 

2 

Thomas 

Williams 

3 

Thompson 

Wilson 

4 

Wi1 son 


5 


0 0 0 0 0 


t 

keys are 


Anderson 
Brown 
Davi s 
Garci a 
Harri s 
Dackson 
Johnson 
Bones 
Marti n 
Marti nez 
Miller 
Moore 
Robinson 
Smith 
Taylor 
Thomas 
Thompson 
Whi te 
Williams 
Wi1 son 


0 0 0 
0 0 
0 0 0 
0 0 


0 0 
0 

0 0 
0 

0 0 
0 
0 

0 0 
0 0 
0 0 
0 


0 0 


small integers 

Typical candidate for key-indexed counting 


Compute frequency counts. The first step is to count 
the frequency of occurrence of each key value, using 
an i nt array count []. For each item, we use the key to 
access an entry in count [] and increment that entry. If 
the key value is r, we increment count [r+1]. (Why+1? 
The reason for that will become clear in the next step.) 
In the example at left, we first increment count[3] 
because Anderson is in section 2, then we increment 
count[4] twice because Brown and Davis are in sec¬ 
tion 3, and so forth. Note that count [0] is always 0, 
and that count [1] is 0 in this example (no students are 
in section 0). 


number of 3s ^ 

Computing frequency counts 
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Transform counts to indices. Next, we use count [] 
to compute, for each key value, the starting index 
positions in the sorted order of items with that key. 
In our example, since there are three items with key 
1 and five items with key 2, then the items with key 
B start at position 8 in the sorted array. In general, 
to get the starting index for items with any given key 
value we sum the frequency counts of smaller val¬ 
ues. For each key value r, the sum of the counts for 
key values less than r+1 is equal to the sum of the 
counts for key values less than r plus count [r], so 
it is easy to proceed from left to right to transform 
count [] into an index table that we can use to sort 
the data. 


for (int r = 0; r < R; r++) 
count[r+l] += count[r]; 


always 0 count [] 


r \ 0 1 2 
0 0 0 3 
10 0 3 
2 0 0 3 



3 4 5 

5 6 6 

5 6 6 

5 6 6 

8 6 
8 14 6 

3 14 20 

8 14 20 


number of keys less than 3 
(start index of 3s in output) 


Transforming counts to start indices 


for (int i = 0; i < N; i++) 

aux[count[a[i] .key()]++] = a[i] ; 


count [] 


i 

1 

2 

3 

4 







0 

0 

3 

8 

14 







1 

0 

4 

8 

14 

a[0] 

Anderson 

2 

Harri s 

1 

aux[0] 

2 

0 

4 

9 

14 

a[l] 

Brown 

3 \ 

Martin 

1 

aux[l] 

3 

0 

4 

10 

14 

a[2] 

Davi s 

3 A 

Moore 

1 

aux[2] 

4 

0 

4 

10 

15 

a[3] 

Garcia 

4 X 

K Anderson 

2 

aux[3] 

5 

1 

4 

10 

15 

a [4] 

Harri s 

1 V 

\Y/ Martinez 

2 

aux[4] 

6 

1 

4 

11 

15 

a[5] 

Jackson 

3 \ 

XV/ Miller 

2 

aux[5] 

7 

1 

4 

11 

16 

a[6] 

Johnson 

4 \ 

£ AV^Robinson 

2 

aux[6] 

8 

1 

4 

12 

16 

a[7] 

Jones 

3 y 

YAXwhi te 

2 

aux[7] 

9 

2 

4 

12 

16 

a [8] 

Martin 

1 A 

Vyv X Brown 

3 

aux [8] 

10 

2 

5 

12 

16 

a[9] 

Martinez 

2 

\\ \ Davi s 

3 

aux[9] 

11 

2 

6 

12 

16 

a [10] 

Miller 

2 

ANXjackson 

3 

aux[10] 

12 

3 

6 

12 

16 

a[ll] 

Moore 

1 / 

Y\ 'Jones 

3 

aux [11] 

13 

3 

7 

12 

16 

a[12] 

Robinson 

2 

/ VuTaylor 

3 

aux[12] 

14 

3 

7 

12 

17 

a[13] 

Smith 

4 x 

/•■'''A yWi 11 i ams 

3 

aux[13] 

15 

3 

7 

13 

17 

a [14] 

Taylor 

3^ 

/ /Garci a 

4 

aux[14] 

16 

3 

7 

13 

18 

a[15] 

Thomas 

4 

X Johnson 

4 

aux[15] 

17 

3 

7 

13 

19 

a[16] 

Thompson 

4 h 

Smith 

4 

aux[16] 

18 

3 

8 

13 

19 

a[17] 

White 

2 / 

Thomas 

4 

aux[17] 

19 

3 

8 

14 

19 

a[18] 

Williams 

3 / 

Thompson 

4 

aux[18] 


3 

8 

14 

20 

a[19] 

Wi1 son 

4 

Wi1 son 

4 

aux[19] 


3 

8 

14 

20 








Distributing the data (records with key 3 highlighted) 


Distribute the data. With the 
count [] array transformed into 
an index table, we accomplish the 
actual sort by moving the items to 
an auxiliary array aux []. We move 
each item to the position in aux[] 
indicated by the count [] entry cor¬ 
responding to its key, and then in¬ 
crement that entry to maintain the 
following invariant for count []: 
for each key value r, count[r] is 
the index of the position in aux [] 
where the next item with key value 
r (if any) should be placed. This 
process produces a sorted result 
with one pass through the data, as 
illustrated at left. Note: In one of 
our applications, the fact that this 
implementation is stable is critical: 
items with equal keys are brought 
together but kept in the same rela¬ 
tive order. 
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before 



t t f t 

count[0] count[1] count[2] count[R-l] 


after 

aux[] 


0 


0 

1 

1 

1 


1 

2 

2 


2 

. . . 

R-l 

R-l 

R-l 


R-l 


\ 

count[0] 


t 

count[1] 


t 

count[2] 


t 

count[R-l] 


Key-indexed counting (distribution phase) 


Copy back. Since we accomplished the sort by moving the items to an auxiliary array, 
the last step is to copy the sorted result back to the original array. 

Proposition A. Key-indexed counting uses 8 N + 3 R + 1 array accesses to stably 
sort N items whose keys are integers between 0 and R — 1. 

Proof: Immediate from the code. Initializing the arrays uses N + R + 1 array ac¬ 
cesses. The first loop increments a counter for each of the N items (2 N array ac¬ 
cesses); the second loop does R additions (2 R array accesses); the third loop does 
N counter increments and Ndata moves (3 N array accesses); and the fourth loop 
does N data moves (2 N array accesses). Both moves preserve the relative order of 
equal keys. 


key-indexed counting is an extremely effective 
and often overlooked sorting method for applica¬ 
tions where keys are small integers. Understand¬ 
ing how it works is a first step toward understand¬ 
ing string sorting, proposition a implies that 
key-indexed counting breaks through the N log 
N lower bound that we proved for sorting. How 
does it manage to do so? proposition i in sec¬ 
tion 2.2 is a lower bound on the number of com¬ 
pares needed (when data is accessed only through 
compareToO)—key-indexed counting does no 
compares (it accesses data only through keyO). 
When R is within a constant factor of N, we have 
a linear-time sort. 


int N = a.length; 

String[] aux = new String[N]; 

int[] count = new int[R+l]; 

// Compute frequency counts. 

for (int i =0; i < N; i++) 
count[a[i] . key() + 1]++; 

// Transform counts to indices. 

for (int r = 0; r < R; r++) 
count[r+l] += count[r]; 

// Distribute the records. 

for (int i =0; i < N; i++) 

aux[count[a[i] .key()]++] = a[i] ; 

// Copy back. 

for (int i =0; i < N; i++) 
a[i] = aux[i] ; 

Key-indexed counting (a[].key is an int in [0, R). 
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LSD String sort The first string-sorting method that we consider is known as least- 
significant-digit first (LSD) string sort. Consider the following motivating application: 
Suppose that a highway engineer sets up a device that records 
the license plate numbers of all vehicles using a busy highway 
for a given period of time and wants to know the number of 
different vehicles that used the highway. As you know from 
section 2.1, one easy way to solve this problem is to sort the 
numbers, then make a pass through to count the different val¬ 
ues, as in Dedu p (page 490). License plates are a mixture of num¬ 
bers and letters, so it is natural to represent them as strings. In 
the simplest situation (such as the California license plate ex¬ 
amples at right) the strings all have the same number of char¬ 
acters. This situation is often found in sort applications—for 
example, telephone numbers, bank account numbers, and IP 
addresses are typically fixed-length strings. 

Sorting such strings can be done with key-indexed count¬ 
ing, as shown in algorithm 5.1 (LSD) and the example be¬ 
low it on the facing page. If the strings are each of length W, 
we sort the strings W times with key-indexed counting, using 

each of the positions as the key, proceeding from right to left. It is not easy, at first, to 
be convinced that the method produces a sorted array—in fact, it does not work at all 
unless the key-indexed count implementation is stable. Keep this fact in mind and refer 
to the example when studying this proof of correctness: 


input 

sorted result 

4PCC9B8 

1ICK750 

2IYE2B0 

1ICK750 

3CI0720 

10HV845 

1ICK750 

10HV845 

10HV845 

10HV845 

4BZY524 

2IYE230 

1ICK750 

2RLA629 

3CI0720 

2RLA629 

10HV845 

3ATW723 

10HV845 

3CI0720 

2RLA629 

3CI0720 

2RLA629 

43ZY524 

3ATW723 

t 

keys are all 
the same length 

4PGC938 


Typical candidate for 
LSD string sort 


Proposition B. LSD string sort stably sorts fixed-length strings. 

Proof: This fact depends crucially on the key-indexed counting implementation 
being stable, as indicated in proposition a. After sorting keys on their i trailing 
characters (in a stable manner), we know that any two keys appear in proper order 
in the array (considering just those characters) either because the first of their i 
trailing characters is different, in which case the sort on that character puts them in 
order, or because the first of their i th trailing characters is the same, in which case 
they are in order because of stability (and by induction, for i -1). 


Another way to state the proof is to think about the futLire: if the characters that have 
not been examined for a pair of keys are identical, any difference between the keys is re¬ 
stricted to the characters already examined, so the keys have been properly ordered and 
will remain so because of stability. If, on the other hand, the characters that have not 
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ALGORITHM 5.1 LSD string sort 


public class LSD 

{ 

public static void sortCString[] a, int W) 

{ // Sort a[] on leading W characters. 

int N = a.length; 
int R = 256; 

String[] aux = new String[N]; 


} 


for (int d = W-l; d >= 0; d--) 

{ // Sort by key-indexed counting on dth char. 

int[] count = new int[R+l]; // Compute frequency counts. 

for (int i =0; i < N; i++) 

count[a[i].charAt(d) + 1]++; 


for (int r = 0; r < R; r++) // Transform counts to indices. 

count[r+l] += count[r]; 


for (int i =0; i < N; i++) 
aux[count[a[i].charAt(d)]++] 


// Distribute. 
= a [ i ]; 


for (int i =0; i < N; i++) // Copy back. 

a[i] = aux[i] ; 

} 


To sort an array a[] of strings that each have exactly W characters, we do W key-indexed counting 
sorts: one for each character position, proceeding from right to left. 


input {W= 7) 

d= 6 

d = 5 

d = 4 

d= 3 

d= 2 

d= 1 

d= 0 

output 

4PGC958 

2IYE230 

[0720 

230 

A629 

CK750 

ATW723 

1ICK750 

1ICK750 

2IYE250 

>CI0720 

[0720 

524 

A629 

CK750 

CI0720 

1ICK750 

1ICK750 

3CI0720 

1ICK750 

3ATW723 

629 

C938 

CC938 

CI0720 

10HV845 

10HV845 

1ICK750 

1ICK750 

43ZY524 

2RLA629 

E230 

HV845 

ICK7 50 

10HV845 

10HV845 

10HV845 

3CI0720 

29 

720 

K750 

HV845 

ICK7 50 

10HV845 

10HV845 

4JZY524 

3ATW723 

29 

720 

K750 

HV845 

IYE230 

2IYE230 

2IYE230 

1ICK750 

43ZY524 

7E2 30 

723 

0720 

10720 

3ZY524 

2RLA629 

2RLA629 

3CI0720 

10HV845 

PCC938 

:i<750 

0720 

10720 

OHV845 

2RLA629 

2RLA629 

10HV845 

10HV845 

0HV845 

:K750 

V845 

LA629 

OHV845 

3ATW723 

3ATW723 

10HV845 

10HV845 

L0HV845 

)HV845 

V845 

LA629 

OHV845 

3CI0720 

3CI0720 

2RLA629 

4PCC938 

0HV845 

)HV845 

V845 

TW72 3 

PCC938 

3CI0720 

3CI0720 

2RLA629 

2RLA629 

:K750 

)HV845 

W72 3 

YE230 

RLA629 

43ZY524 

43ZY524 

3ATW723 

2RLA629 

LICK750 

4PCC938 

4JZY524 

ZY524 

RLA629 

4PGC938 

4PGC938 
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A3 

♦ A 

A A 

¥ 6 

¥A 

A 2 

♦ A 

AA 

A3 

¥ A 

A A 

A 4 

AK 

A 2 

A 5 

*] 

A 2 

A 6 

♦ Q 

¥2 

A 7 

A 6 

♦ 2 

A 8 

A 3 

¥ 3 

A9 

AA 

A3 

A10 

♦ 9 

A3 

A3 

¥ 9 

♦ 3 

AQ 

♦ 8 

♦ 4 

AK 

A9 

A 4 

¥ A 

AK 

¥4 

¥2 

♦ 4 

A 4 

¥ 3 

A 5 

A 5 

¥ 4 

AQ 

♦ 5 

¥ 5 

¥ 3 

A 5 

¥ 6 

A 2 

¥ 5 

¥7 

A10 

♦ 6 

¥8 

A 9 

A 6 

¥9 

¥7 

A 6 

V10 

A4 

♦ 6 

¥ 3 

¥ 4 

¥7 

¥ Q 

♦ 10 

A7 

¥ K 

A A 

A 7 

♦ A 

♦ 5 

♦ 7 

♦ 2 

A3 

♦ 8 

♦ 3 

¥8 

♦ 8 

♦ 4 

A 2 

A 8 

♦ 5 

♦ K 

A 8 

♦ 6 

A 4 

♦ 9 

♦ 7 

A 7 

¥9 

♦ 8 

¥Q 

A 9 

♦ 9 

♦ 3 

A 9 

♦ 10 

A 6 

A10 

♦ 3 

A3 

♦ 10 

♦ Q 

A 7 

A10 

♦ K 

A 8 

V10 

AA 

A10 

A3 

A 2 

♦ 3 

¥ 3 

A3 

V10 

A3 

A 4 

♦ 7 

♦ 3 

A 5 

AQ 

♦ Q 

A 6 

¥2 

AQ 

A 7 

♦ 2 

¥Q 

A 8 

A 5 

AQ 

A 9 

¥ K 

AK 

A10 

¥ 5 

AK 

A3 

♦ 6 

♦ K 

AQ 

A 8 

¥ K 

AK 


Sorting a card deck with 
LSD string sort 


■ Strings 

been examined are different, the characters already examined do not matter, 
and a later pass will correctly order the pair based on the more significant 
differences. 

LSD radix sorting is the method used by the old punched-card-sorting 
machines that were developed at the beginning of the 20th century and thus 
predated the use of computers in commercial data processing by several de¬ 
cades. Such machines had the capability of distributing a deck of punched 
cards among 10 bins, according to the pattern of holes punched in the select¬ 
ed columns. If a deck of cards had numbers punched in a particular set of 
columns, an operator could sort the cards by running them through the ma¬ 
chine on the rightmost digit, then picking up and stacking the output decks 
in order, then running them through the machine on the next-to-rightmost 
digit, and so forth, until getting to the first digit. The physical stacking of the 
cards is a stable process, which is mimicked by key-indexed counting sort. 
Not only was this version of LSD radix sorting important in commercial 
applications up through the 1970s, blit it was also used by many cautious 
programmers (and students!), who would have to keep their programs on 
punched cards (one line per card) and would punch sequence numbers in 
the final few columns of a program deck so as to be able to put the deck back 
in order mechanically if it were accidentally dropped. This method is also a 
neat way to sort a deck of playing cards: deal them into thirteen piles (one for 
each value), pick up the piles in order, then deal into four piles (one for each 
suit). The (stable) dealing process keeps the cards in order within each suit, 
so picking up the piles in suit order yields a sorted deck. 

In many string-sorting applications (even license plates, for some states), 
the keys are not all be the same length. It is possible to adapt LSD string sort 
to work for such applications, but we leave this task for exercises because 
we will next consider two other methods that are specifically designed for 
variable-length keys. 

From a theoretical standpoint, LSD string sort is significant because it is a 
linear -time sort for typical applications. No matter how large the value of N, 
it makes W passes through the data. Specifically: 
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Proposition B (continued). LSD string sort uses ~7WN + 3 WR array accesses and 
extra space proportional to N+ R to sort N items whose keys are W-c haracter 
strings taken from an /^-character alphabet. 

Proof: The method is W passes of key-indexed counting, except that the aux [] ar¬ 
ray is initialized just once. The total is immediate from the code and proposition a. 


For typical applications, R is far smaller than N, so proposition b implies that the to¬ 
tal running time is proportional to WN. An input array of N strings that each have W 
characters has a total of WN characters, so the running time of LSD string sort is linear 
in the size of the input. 
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Sorting a card deck with 
MSD string sort 


MSD string sort To implement a general-purpose string sort, where 
strings are not necessarily all the same length, we consider the characters 
in left-to-right order. We know that strings that start with a should appear 
before strings that start with b, and so forth. The natural way to implement 
this idea is a recursive method known as most-significant-digit-first (MSD) 
string sort. We use key-indexed counting to sort the strings according to 
their first character, then (recursively) sort the subarrays corresponding to 
each character (excluding the first character, which we know to be the same 
for each string in each subarray). Like quicksort, MSD string sort parti¬ 
tions the array into subarrays that can be sorted independently to complete 
the job, but it partitions the array into one subarray for each possible value 
of the first character, instead of the two or three partitions in quicksort. 

End-of-string convention. We need to pay particular attention to reaching 
the ends of strings in MSD string sort. For a proper sort, we need the sub¬ 
array for strings whose charac- 

ters have all been examined to sort on first character value recursively sort subarrays 

r . ^ i to partition into subarrays (excluding first character) 

appear as the first subarray, and 
we do not want to recursively 
sort this subarray. To facilitate 
these two parts of the compu¬ 
tation we use a private two- 
argument toChar () method to 
convert from an indexed string 
character to an array index 
that returns -1 if the specified 
character position is past the 
end of the string. Then, we just 
add 1 to each returned value, to 
get a nonnegative i nt that we 
can use to index count []. This 
convention means that we have 
R+l different possible charac¬ 
ter values at each string posi¬ 
tion: 0 to signify end of string, 

1 for the first alphabet charac¬ 
ter, 2 for the second alphabet 
character, and so forth. Since 

Overview of MSD string sort 
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key-indexed counting already needs one extra position, we use the 
code int count [] = new int[R+2]; to create the array of fre¬ 
quency counts (and set all of its values to 0). Note: Some languages, 
notably C and C++, have a built-in end-of-string convention, so 
our code needs to be adjusted accordingly for such languages. 

with these preparations, the implementation of MSD string 
sort, in algorithm 5 . 2 , requires very little new code. We add a test 
to cutoff to insertion sort for small subarrays (using a specialized 
insertion sort that we will consider later), and we add a loop to 
key-indexed counting to do the recursive calls. As summarized in 
the table at the bottom of this page, the values in the count [] array 
(after serving to count the frequencies, transform counts to indices, 
and distribute the data) give us precisely the information that we 
need to (recursively) sort the subarrays corresponding to each character value. 


input 

sorted result 

she 

are 

sel 1 s 

by 

seashel1s 

seashel1s 

by 

seashel1s 

the 

seashore 

seashore 

sel 1 s 

the 

sel 1 s 

shells various 

she 

she-— yj key 

she 

sells'?^ 5 

shel1s 

aPe / 

surely 

surely 

the 

seashel1s 

the 

Typical candidate for MSD string sort 


Specified alphabet. The cost of MSD string sort depends strongly on the number of 
possible characters in the alphabet. It is easy to modify our sort method to take an 
Alphabet as argument, to allow for improved efficiency in clients involving strings 
taken from relatively small alphabets. The following changes will do the job: 

■ Save the alphabet in an instance variable al pha in the constructor. 

■ Set R to al pha. R() in the constructor. 

■ Replace s.charAt(d) with alpha.toIndex(s.charAt(d)) in charAtQ. 


at completion 
of phase for 
dth character 

value of count [r] is 

r = 0 

r = 1 

r between 2 and R-1 

r = R 

r = R+l 

count 

frequencies 

0 (not used) 

number of 
strings of 
length d 

number of strings whose 
dth character value is r-2 

transform 
counts to 

indices 

start index of subarray for 
strings of length d 

start index of subarray for strings whose 
dth character value is r-1 

not 

used 

distribute 

start index of subarray for strings 
whose dth character value is r 

not 

used 

1 + end index of subarray 
for strings of length d 

1 + end index of subarray for strings 
whose dth character value is r-1 

not 

used 


Interpretation of count[] values during MSD string sort 
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ALGORITHM 5.2 MSD string sort 


public class MSD 

{ 

private static int R = 256; // radix 

private static final int M = 15; // cutoff for small subarrays 

private static String[] aux; // auxiliary array for distribution 

private static int charAt(String s, int d) 

{ if (d < s.lengthO) return s.charAt(d); else return -1; } 

public static void sortCString[] a) 

{ 

int N = a.length; 
aux = new String[N]; 
sort(a, 0, N-l, 0); 

} 

private static void sort(String[] a, int lo, int hi, int d) 

{ // Sort from a[lo] to a[hi], starting at the dth character. 

if (hi <= lo + M) 

{ Insertion.sort(a, lo, hi, d); return; } 

int[] count = new int[R+2]; // Compute frequency counts. 

for (int i = lo; i <= hi; i++) 
count[charAt(a[i], d) + 2]++; 

for (int r = 0; r < R+l; r++) // Transform counts to indices. 

count[r+l] += count[r]; 

for (int i = lo; i <= hi; i++) // Distribute. 

aux[count[charAt(a[i], d) + 1]++] = a[i]; 

for (int i = lo; i <= hi; i++) // Copy back. 

a[i] = aux[i - lo] ; 

// Recursively sort for each character value. 

for (int r = 0; r < R; r++) 

sort(a, lo + count[r], lo + count[r+l] - 1, d+1); 


To sort an array a [] of strings, we sort them on their first character using key-indexed counting, then 
(recursively) sort the subarrays corresponding to each first-character value. 
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In our running examples, we use strings made up of lowercase letters. It is also easy to 
extend LSD string sort to provide this feature, but typically with much less impact on 
performance than for MSD string sort. 


the code in algorithm 5.2 is deceptively simple, masking a rather sophisticated 
computation. It is definitely worth your while to study the trace of the top level at 
the bottom of this page and the trace of recursive calls on the next page, to be sure 
that you understand the intricacies of the algorithm. This trace uses a cutoff-for-small- 
subarrays threshold value (M) of 0, so that you can see the sort to completion for this 
small example. The strings in this example are taken from A1 phabet. LOWERCASE, with 
R = 26; bear in mind that typical applications might use A1 phabet. EXTENDED. ASCII, 
with R = 256, or A1 phabet.UNICODE, with R = 65536. For large alphabets, MSD 
string sort is so simple as to be dangerous—improperly used, it can consume outra¬ 
geous amounts of time and space. Before considering performance characteristics in 
detail, we shall discuss three important issues (all of which we have considered before, 
in chapter 2) that must be addressed in any application. 

Small subarrays. The basic idea behind MSD string sort is quite effective: in typi¬ 
cal applications, the strings will be in order after examining only a few characters in 
the key. Put another way, the method quickly divides the array to be sorted into small 
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Trace of recursive calls for MSD string sort (no cutoff for small subarrays, subarrays of size 0 and 1 omitted) 


subarrays. But this is a double-edged sword: we are certain to have to handle huge num¬ 
bers of tiny subarrays, so we had better be sure that we handle them efficiently Small 
subarrays are of critical importance in the performance of MSD string sort. We have seen 
this situation for other recursive sorts (quicksort and mergesort), but it is much more 
dramatic for MSD string sort. For example, suppose that you are sorting millions of 
ASCII strings (R = 256) that are all different, with no cutoff for small subarrays. Each 
string eventually finds its way to its own subarray, so you will sort millions of subarrays 
of size 1. But each such sort involves initializing the 258 entries of the count [] array 
to 0 and transforming them all to indices. This cost is likely to dominate the rest of the 
sort. With Unicode (R = 65536) the sort might be thousands of times slower. Indeed, 
many unsuspecting sort clients have seen their running times explode from minutes to 
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public static void sort(String[] a, int lo, int hi, int d) 

{ // Sort from a[lo] to a[hi], starting at the dth character. 


} 


for (int i = lo; i <= hi; i++) 

for (int j = i; j > lo && less(a[j], a [ j-1] , d) ; j—) 
exch(a, j, j-1); 


private static boolean less(String v, String w, int d) 

{ return v.substring(d).compareTo(w.substring(d)) <0; } 

Insertion sort for strings whose first d characters are equal 


100 % ■ 


hours on switching from ASCII to Unicode, for precisely this reason. Accordingly, the 
switch to insertion sort for small subarrays is a must for MSD string sort. To avoid the 
cost of reexamining characters that we know to be equal, we use the version of insertion 
sort given at the top of the page, which takes an extra argument d and assumes that the 
first d characters of all the strings to be sorted are known to be equal. The efficiency of 
this code depends on substri ng() being a constant-time 
operation. As with quicksort and mergesort, most of the 
benefit of this improvement is achieved with a small value 
of the cutoff, but the savings here are much more dramat¬ 
ic. The diagram at right shows the results of experiments 
where using a cutoff to insertion sort for subarrays of size 
10 or less decreases the running time by a factor of 10 for 
a typical application. 

Equal keys. A second pitfall for MSD string sort is that 
it can be relatively slow for subarrays containing large 
numbers of equal keys. If a substring occurs sufficiently 
often that the cutoff for small subarrays does not ap¬ 
ply, then a recursive call is needed for every character 
in all of the equal keys. Moreover, key-indexed count¬ 
ing is an inefficient way to determine that the charac¬ 
ters are all equal: not only does each character need to 
be examined and each string moved, but all the counts 
have to be initialized, converted to indices, and so forth. 

Thus, the worst case for MSD string sorting is when all 
keys are equal. The same problem arises when large num¬ 
bers of keys have long common prefixes, a situation often 
found in applications. 
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Extra space. To do the partitioning, MSD uses two auxiliary arrays: the temporary 
array for distributing keys (aux []) and the array that holds the counts that are trans¬ 
formed into partition indices (count []). The aux [] array is of size N and can be cre¬ 
ated outside the recursive sort() method. This extra space can be eliminated by sac¬ 
rificing stability (see exercise 5 . 1 . 17 ), but it is often not a major concern in practical 
applications of MSD string sort. Space for the count [] array, on the other hand, can be 
an important issue (because it cannot be created outside the recursive sortO method) 
as addressed in proposition d below. 


Random string model. To study the performance of MSD string sort, we use a random 
string model, where each string consists of (independently) random characters, with 
no bound on their length. Long equal keys are 
essentially ignored, because they are extremely 
unlikely. The behavior of MSD string sort in 
this model is similar to its behavior in a model 
where we consider random fixed-length keys 
and also to its performance for typical real 
data; in all three, MSD string sort tends to ex¬ 
amine just a few characters at the beginning of 
each key, as we will see. 

Performance. The running time of MSD 
string sort depends on the data. For compare- 
based methods, we were primarily concerned 
with the order of the keys; for MSD string sort, 
the order of the keys is immaterial, but we are 
concerned with the values of the keys. 

■ For random inputs, MSD string sort 

examines just enough characters to distinguish among the keys, and the running 

time is sublinear in the number of characters in the data (it examines a small 
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Characters examined by MSD string sort 


fraction of the input characters). 

■ For nonrandom inputs, MSD string sort still could be sublinear but might need 
to examine more characters than in the random case, depending on the data. In 
particular, it has to examine all the characters in equla keys, so the running time 
is nearly linear in the number of characters in the data when significant num¬ 
bers of equal keys are present. 

■ In the worst case, MSD string sort examines all the characters in the keys, so the 
running time is linear in the number of characters in the data (like LSD string 
sort). A worst-case input is one with all strings equal. 
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Some applications involve distinct keys that are well-modeled by the random string 
model; others have significant numbers of equal keys or long common prefixes, so the 
sort time is closer to the worst case. Our license-plate-processing application, for ex¬ 
ample, can fall anywhere between these extremes: if our engineer takes an hour of data 
from a busy interstate, there will not be many duplicates and the random model will 
apply; for a week’s worth of data on a local road, there will be numerous duplicates and 
performance will be closer to the worst case. 

Proposition C. To sort N random strings from an R-character alphabet, MSD 
string sort examines about N log R N characters, on average. 

Proof sketch: We expect the subarrays to be all about the same size, so the recur¬ 
rence C N = RC n/r + N approximately describes the performance, which leads to the 
stated result, generalizing our argument for quicksort in chapter 2 . Again, this 
description of the situation is not entirely accurate, because N/R is not necessarily 
an integer, and the subarrays are the same size only on the average (and because 
the number of characters in real keys is finite). These effects turn out to be less 
significant for MSD string sort than for standard quicksort, so the leading term of 
the running time is the solution to this recurrence. The detailed analysis that proves 
this fact is a classical example in the analysis of algorithms, first done by Knuth in 
the early 1970s. 


As food for thought and to indicate why the proof is beyond the scope of this book, 
note that key length does not play a role. Indeed, the random-string model allows key 
length to approach infinity. There is a nonzero probability that two keys will match for 
any specified number of characters, but this probability is so small as to not play a role 
in our performance estimates. 

As we have discussed, the number of characters examined is not the full story for 
MSD string sort. We also have to take into account the time and space required to count 
frequencies and turn the counts into indices. 

Proposition D. MSD string sort uses between 8 N + 3 R and ~7wN + 3 WR array ac¬ 
cesses to sort N strings taken from an /^-character alphabet, where w is the average 
string length. 

Proof: Immediate from the code, proposition a, and proposition b. In the best 
case MSD sort uses just one pass; in the worst case, it performs like LSD string sort. 
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When N is small, the factor of R dominates. Though precise analysis of the total cost 
becomes difficult and complicated, you can estimate the effect of this cost just by con¬ 
sidering small subarrays when keys are distinct. With no cutoff for small subarrays, 
each key appears in its own subarray, so NR array accesses are needed for just these 
subarrays. If we cut off to small subarrays of size M, we have about N/M subarrays of 
size M, so we are trading off NR/M array accesses with NMI 4 compares, which tells us 
that we should choose M to be proportional to the square root of R. 

Proposition D (continued). To sort N strings taken from an R-character alphabet, 
the amount of space needed by MSD string sort is proportional to R times the 
length of the longest string (plus N), in the worst case. 

Proof: The count [] array must be created within sort(), so the total amount of 
space needed is proportional to R times the depth of recursion (plus N for the aux¬ 
iliary array). Precisely, the depth of the recursion is the length of the longest string 
that is a prefix of two or more of the strings to be sorted. 


As just discussed, equal keys cause the depth of the recursion to be proportional to the 
length of the keys. The immediate practical lesson to be drawn from proposition d 
is that it is quite possible for MSD string sort to run out of time or space when sorting 
long strings taken from large alphabets, particularly if long equal keys are to be ex¬ 
pected. For example, with A1 phabet. UNICODE and more than M equal 1,000-character 
strings, MSD. sort () would require space for over 65 million counters! 

the main challenge in getting maximum efficiency from MSD string sort on keys 
that are long strings is to deal with lack of randomness in the data. Typically, keys may 
have long stretches of equal data, or parts of them might fall in only a narrow range. 
For example, an information-processing application for student data might have keys 
that include graduation year (4 bytes, but one of four different values), state names 
(perhaps 10 bytes, but one of 50 different values), and gender (1 byte with one of two 
given values), as well as a person’s name (more similar to random strings, but probably 
not short, with nonuniform letter distributions, and with trailing blanks in a fixed- 
length field). Restrictions like these lead to large numbers of empty subarrays during 
the MSD string sort. Next, we consider a graceful way to adapt to such situations. 
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use first character value 
to partition into “less,” “equal,” 
and “greater” subarrays 


Three-way string quicksort We can also 
adapt quicksort to MSD string sorting by using 
3-way partitioning on the leading character of the 
keys, moving to the next character on only the mid¬ 
dle subarray (keys with leading character equal to 
the partitioning character). This method is not diffi¬ 
cult to implement, as you can see in algorithm 5 . 3 : 
we just add an argument to the recursive method 
in algorithm 2.5 that keeps track of the current 
character, adapt the 3-way partitioning code to use 
that character, and appropriately modify the recur¬ 
sive calls. 

Although it does the computation in a different 
order, 3-way string quicksort amounts to sorting the 
array on the leading characters of the keys (using 
quicksort), then applying the method recursively on 
the remainder of the keys. For sorting strings, the 
method compares favorably with normal quicksort 
and with MSD string sort. Indeed, it is a hybrid of 
these two algorithms. 

Three-way string quicksort divides the array into 
only three parts, so it involves more data movement 

than MSD string sort when the number of nonempty partitions is large because it has to 

do a series of 3-way partitions 
to get the effect of the multiway 
partition. On the other hand, 
MSD string sort can create 
large numbers of (empty) sub¬ 
arrays, whereas 3-way string 
quicksort always has just three. 
Thus, 3-way string quicksort 
adapts well to handling equal 
keys, keys with long common 
prefixes, keys that fall into a 
small range, and small arrays— 
all situations where MSD 
string sort runs slowly. Of par¬ 
ticular importance is that the 
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ALGORITHM 5.3 Three-way string quicksort 


public class Quick3string 

{ 

private static int charAt(String s, int d) 

{ if (d < s.lengthO) return s.charAt(d); else return -1; } 

public static void sortCString[] a) 

{ sort(a, 0, a.length - 1, 0); } 

private static void sort(String[] a, int lo, int hi, int d) 

{ 

if (hi <= lo) return; 

int It = lo, gt = hi; 
int v = charAt(a[lo], d); 
int i = lo + 1; 
while (i <= gt) 

{ 



int t 


charAt(a[i ], d) ; 

i 



if 


Ct < v) 

exch(a, 

11++, 

i ++); 


el se 

if 

Ct > v) 

exch(a, 

i , gt- 

-); 

} 

el se 



i++; 



// 

a[lo. 

.It 

-1] < V 

= a[lt. . 

■ gt] < 

a[gt+l 

sort(a, 

lo, 

lt-1, i 

d); 



if 

(v >= 

0) 

sort(a 

, It, gt, 

, d+1) ; 



sort(a, gt+1, hi, d); 

} 

} 


To sort an array a [] of strings, we 3-way partition them on their first character, then (recursively) sort 
the three resulting subarrays: the strings whose first character is less than the partitioning character, 
the strings whose first character is equal to the partitioning character (excluding their first character 
in the sort), and the strings whose first character is greater than the partitioning character. 
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partitioning adapts to different kinds of structure in different parts of the key. Also, like 
quicksort, 3-way string quicksort does not use extra space (other than the implicit stack 
to support recursion), which is an important advantage over MSD string sort, which 
requires space for both frequency counts and an auxiliary array. 

The figure at the bottom ofthis page shows all ofthe recursive calls that Qui ck3stri ng 
makes for our example. Each subarray is sorted using precisely three recursive calls, ex¬ 
cept when we skip the recursive call on reaching the ends of the (equal) string(s) in the 
middle subarray. 

As usual, in practice, it is worthwhile to consider various standard improvements to 
the implementation in algorithm 5 . 3 : 

Small subarrays. In any recursive algorithm, we can gain efficiency by treating small 
subarrays differently. In this case, we use the insertion sort from page 715, which skips the 
characters that are known to be equal. The improvement due to this change is likely to 
be significant, though not nearly as important as for MSD string sort. 

Restricted alphabet. To handle specialized alphabets, we could add an Alphabet 
argument alpha to each of the methods and replace s.charAt(d) with 
alpha. toIndex(s.charAt(d)) in charAtf). In this case, there is no benefit to doing 
so, and adding this code is likely to substantially slow the algorithm down because this 
code is in the inner loop. 
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Trace of recursive calls for 3-way string quicksort (no cutoff for small subarrays) 
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Randomization. As with any quicksort, it is generally worthwhile to shuffle the array 
beforehand or to use a random paritioning item by swapping the first item with a ran¬ 
dom one. The primary reason to do so is to protect against worst-case performance in 
the case that the array is already sorted or nearly sorted. 

For string keys, standard quicksort and all the other sorts in chapter 2 are actually 
MSD string sorts, because the compareToO method in String accesses the charac¬ 
ters in left-to-right order. That is, compareToO accesses only the leading characters 
if they are different, the leading two characters if the first characters are the same and 
the second different, and so forth. For example, if the first characters of the strings are 
all different, the standard sorts will examine just those characters, thus automatically 
realizing some of the same performance gain that we seek in MSD string sorting. The 
essential idea behind 3-way quicksort is to take special action when the leading char¬ 
acters are equal. Indeed, one way to think of algorithm 5.3 is as a way for standard 
quicksort to keep track of leading characters that are known to be equal. In the small 
subarrays, where most of the compares in the sort are done, the strings are likely to have 
numerous equal leading characters. The standard algorithm has to scan over all those 
characters for each compare; the 3-way algorithm avoids doing so. 

Performance. Consider a case where the string keys are long (and are all the same 
length, for simplicity), but most of the leading characters are equal. In such a situa¬ 
tion, the running time of standard quicksort is proportional to the string length times 
2N\n N, whereas the running time of 3-way string quicksort is proportional to N times 
the string length (to discover all the leading equal characters) plus 2N In N character 
comparisons (to do the sort on the remaining short keys). That is, 3-way string quick¬ 
sort requires up to a factor of 2 lniV fewer character compares than normal quicksort. 
It is not unusual for keys in practical sorting applications to have characteristics similar 
to this artificial example. 
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Proposition E. To sort an array of N random strings, 3-way string quicksort uses 
~ 2/Vln N character compares, on the average. 

Proof: There are two instructive ways to understand this result. First, consider¬ 
ing the method to be equivalent to quicksort partitioning on the leading char¬ 
acter, then (recursively) using the same method on the subarrays, we should not 
be surprised that the total number of operations is about the same as for normal 
quicksort—but they are single-character compares, not full-key compares. Second, 
considering the method as replacing key-indexed counting by quicksort, we expect 
that the N\og R N running time from proposition d should be multiplied by a fac¬ 
tor of 2 In R because it takes quicksort 2 R In R steps to sort R characters, as opposed 
to R steps for the same characters in the MSD string sort. We omit the full proof. 


As emphasized on page 716, considering random strings is instructive, but more detailed 
analysis is needed to predict performance for practical situations. Researchers have 
studied this algorithm in depth and have proved that no algorithm can beat 3-way 
string quicksort (measured by number of character compares) by more than a constant 
factor, under very general assumptions. To appreciate its versatility, note that 3-way 
string quicksort has no direct dependencies on the size of the alphabet. 

Example: web logs. As an example where 3-way string quicksort shines, we can con¬ 
sider a typical modern data-processing task. Suppose that you have built a website and 
want to analyze the traffic that it generates. You can have your system administrator 
supply you with a web log of all transactions on your site. Among the information asso¬ 
ciated with a transaction is the domain name of the originating machine. For example, 
the file week. 1 og. txt on the booksite is a log of one week’s transactions on our book- 
site. Why does 3-way string quicksort do well on such a file? Because the sorted result is 
replete with long common prefixes that this method does not have to reexamine. 
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Which string-sorting algorithm should I use? Naturally, we are interested 
in how the string-sorting methods that we have considered compare to the general- 
purpose methods that we considered in chapter 2 . The following table summarizes 
the important characteristics of the string-sort algorithms that we have discussed in 
this section (the rows for quicksort, mergesort, and 3-way quicksort are included from 
chapter 2 , for comparison). 

order of growth of 


sweet spot 


small arrays, 
arrays in order 

general-purpose 
when space is tight 

general-purpose 
stable sort 

large numbers of 
equal keys 

short fixed-length 
strings 

random strings 

general-purpose, 
strings with long 
prefix matches 


algorithm 

stable? 

inplace? 

typical number calls to charAtO 
to sort N strings 
from an /?-character alphabet 
(average length w, max length I/I/) 




running time 

extra space 

insertion sort 
for strings 

yes 

yes 

between 

N and N 2 

1 

quicksort 

no 

yes 

N log 2 N 

log N 

mergesort 

yes 

no 

N log 2 N 

N 

3-way quicksort 

no 

yes 

between 

N and N log N 

log N 

LSD string sort 

yes 

no 

NW 

N 

MSD string sort 

yes 

no 

between 

N and Nw 

N+ WR 

3-way string 
quicksort 

no 

yes 

between 

N and Nw 

W + log N 


Performance characteristics of string-sorting algorithms 


As in chapter 2 , multiplying these growth rates by appropriate algorithm- and data- 
dependent constants gives an effective way to predict running time. 

As explored in the examples that we have already considered and in many other 
examples in the exercises, different specific situations call for different methods, with 
appropriate parameter settings. In the hands of an expert (maybe that’s you, by now), 
dramatic savings can be realized for certain situations. 
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| Q&A 


Q. Does the Java system sort use one of these methods for Stri ng sorts? 

A. No, but the standard implementation includes a fast string compare that makes 
standard sorts competitive with the methods considered here. 

Q. So, I should just use the system sort for Stri ng keys? 

A. Probably yes in Java, though if you have huge numbers of strings or need an excep¬ 
tionally fast sort, you may wish to switch to char arrays instead of Stri ng values and 
use a radix sort. 

Q. What is explanation of the log 2 N factors on the table in the previous page? 

A. They reflect the idea that most of the comparisons for these algorithms wind up 
being between keys with a common prefix of length log N. Recent research has estab¬ 
lished this fact for random strings with careful mathematical analysis (see booksite for 
reference). 
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EXERCISES 


5 . 1.1 Develop a sort implementation that counts the number of different key values, 
then uses a symbol table to apply key-indexed counting to sort the array. (This method 
is not for use when the number of different key values is large.) 

5 . 1.2 Give a trace for LSD string sort for the keys 

no is th ti fo al go pe to co to th ai of th pa 

5 . 1.3 Give a trace for MSD string sort for the keys 

no is th ti fo al go pe to co to th ai of th pa 

5 . 1.4 Give a trace for 3-way string quicksort for the keys 

no is th ti fo al go pe to co to th ai of th pa 

5 . 1.5 Give a trace for MSD string sort for the keys 

now is the time for all good people to come to the aid of 

5 . 1.6 Give a trace for 3-way string quicksort for the keys 

now is the time for all good people to come to the aid of 

5 . 1.7 Develop an implementation of key-indexed counting that makes use of an array 
of Queue objects. 

5 . 1.8 Give the number of characters examined by MSD string sort and 3-way string 
quicksort for a file of N keys a, aa, aaa, aaaa, aaaaa,... 

5 . 1.9 Develop an implementation of LSD string sort that works for variable-length 
strings. 

5 . 1.10 What is the total number of characters examined by 3-way string quicksort 
when sorting N fixed-length strings (all of length W), in the worst case? 
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CREATIVE PROBLEMS 


5 . 1.11 Queue sort. Implement MSD string sorting using queues, as follows: Keep one 
queue for each bin. On a first pass through the items to be sorted, insert each item into 
the appropriate queue, according to its leading character value. Then, sort the sublists 
and stitch together all the queues to make a sorted whole. Note that this method does 
not involve keeping the count [] arrays within the recursive method. 

5 . 1.12 Alphabet. Develop an implementation of the Alphabet API that is given on 
page 698 and use it to develop LSD and MSD sorts for general alphabets. 

5 . 1.13 Hybrid sort. Investigate the idea of using standard MSD string sort for large ar¬ 
rays, in order to get the advantage of multiway partitioning, and 3-way string quicksort 
for smaller arrays, in order to avoid the negative effects of large numbers of empty bins. 

5 . 1.14 Array sort. Develop a method that uses 3-way string quicksort for keys that are 
arrays of i nt values. 

5 . 1.15 Sublinear sort. Develop a sort implementation for i nt values that makes two 
passes through the array to do an LSD sort on the leading 16 bits of the keys, then does 
an insertion sort. 

5 . 1 . 16 Linked-list sort. Develop a sort implementation that takes a linked list of nodes 
with Stri ng key values as argument and rearranges the nodes so that they appear in 
sorted order (returning a link to the node with the smallest key). Use 3-way string 
quicksort. 

5.1.17 In-place key-indexed counting. Develop a version of key-indexed counting that 
uses only a constant amount of extra space. Prove that your version is stable or provide 
a counterexample. 
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EXPERIMENTS 

5.1.18 Random decimal keys. Write a static method randomDecimalKeys that takes 
i nt values N and W as arguments and returns an array of N string values that are each 
W-digit decimal numbers. 

5.1.19 Random CA license plates. Write a static method randomPl atesCA that takes 
an i nt value N as argument and returns an array of N Stri ng values that represent CA 
license plates as in the examples in this section. 

5.1.20 Random fixed-length words. Write a static method randomFixedLengthWords 
that takes i nt values N and W as arguments and returns an array of N string values that 
are each strings of W characters from the alphabet. 

5.1.21 Random items. Write a static method randomltems that takes an i nt value N as 
argument and returns an array of N string values that are each strings of length between 
15 and 30 made up of three fields: a 4-character field with one of a set of 10 fixed strings; 
a 10-char field with one of a set of 50 fixed strings; a 1-character field with one of two 
given values; and a 15-byte field with random left-justified strings of letters equally 
likely to be 4 through 15 characters long. 

5.1.22 Timings. Compare the running times of MSD string sort and 3-way string 
quicksort, using various key generators. For fixed-length keys, include LSD string sort. 

5.1.23 Array accesses. Compare the number of array accesses used by MSD string sort 
and 3-way string sort, using various key generators. For fixed-length keys, include LSD 
string sort. 

5.1.24 Rightmost character accessed. Compare the position of the rightmost character 
accessed for MSD string sort and 3-way string quicksort, using various key generators. 
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As with sorting, we can take advantage of properties of strings to develop search meth¬ 
ods (symbol-table implementations) that can be more efficient than the general-pur¬ 
pose methods of chapter 3 for typical applications where search keys are strings. 

Specifically, the methods that we consider in this section achieve the following per¬ 
formance characteristics in typical applications, even for huge tables: 

■ Search hits take time proportional to the length of the search key. 

■ Search misses involve examining only a few characters. 

On reflection, these performance characteristics are quite remarkable, one of the 
crowning achievements of algorithmic technology and a primary factor in enabling 
the development of the computational infrastructure we now enjoy that has made so 
much information instantly accessible. Moreover, we can extend the symbol-table API 
to include character-basedoperations defined for string keys (but not necessarily for 
all Comparabl e types of keys) that are powerful and quite useful in practice, as in the 
following API: 


public class StringST<Value> 



StringST O 

create a symbol table 

voi d 

putfString key, Value val) 

put key-value pair into the table 
(remove key if value is null) 

Val ue 

getCString key) 

value paired with key 
(null if key is absent) 

voi d 

delete(String key) 

remove key (and its value) 

boolean 

contains(Stri ng key) 

is there a value paired with key? 

boolean 

i sEmptyO 

is the table empty? 

String 

longestPrefixOf(String s) 

the longest key that is a prefix of : 

Iterable<Stri ng> 

keysWithPrefix(String s) 

all the keys having s as a prefix 

Iterable<Stri ng> 

keysThatMatch(String s) 

all the keys that match s 
(where . matches any character) 

int 

size() 

number of key-value pairs 

Iterable<Stri ng> 

keys() 

all the keys in the table 


API for a symbol table with string keys 
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This API differs from the symbol-table API introduced in chapter 3 in the follow¬ 
ing aspects: 

■ We replace the generic type Key with the concrete type String. 

■ We add three new methods, longestPrefixOf (), keysWi thPrefi x() and 
keysThatMatchO. 

We retain the basic conventions of our symbol-table implementations in chapter 3 
(no duplicate or null keys and no null values). 

As we saw for sorting with string keys, it is often quite important to be able to work 
with strings from a specified alphabet. Simple and efficient implementations that are 
the method of choice for small alphabets turn out to be useless for large alphabets be¬ 
cause they consume too much space. In such cases, it is certainly worthwhile to add a 
constructor that allows clients to specify the alphabet. We will consider the implemen¬ 
tation of such a constructor later in this section but omit it from the API for now, in 
order to concentrate on string keys. 

The following descriptions of the three new methods use the keys she sel 1 s sea 
shells by the sea shore to give examples: 

■ 1 ongestPrefi xOf () takes a string as argument and returns the longest 
key in the symbol table that is a prefix of that string. For the keys above, 
longestPrefixOf("shell") is she and longestPrefixOf("shellsort") is 
shel 1 s. 

■ keysWi thPrefi x() takes a string as argument and returns all the keys 
in the symbol table having that string as prefix. For the keys above, 

keysWi thPrefi x("she") is she and shel 1 s, and keysWi thPrefix("se") is 
sel 1 s and sea. 

■ keysThatMatch () takes a string as argument and returns all the keys in the 
symbol table that match that string, in the sense that a period (.) in the argu¬ 
ment string matches any character. For the keys above, keysThatMatch (" . he") 
returns she and the, and keysThatMatch("s . . ") returns she and sea. 

We will consider in detail implementations and applications of these operations after 
we have seen the basic symbol-table methods. These particular operations are repre¬ 
sentative of what is possible with string keys; we discuss several other possibilities in 
the exercises. 

To focus on the main ideas, we concentrate on put(), get(), and the new methods; 
we assume (as in chapter 3 ) default implementations of contai ns() and i sEmptyO; 
and we leave implementations of si ze () and del ete O for exercises. Since strings are 
Comparable, extending the API to also include the ordered operations defined in the 
ordered symbol-table API in chapter 3 is possible (and worthwhile); we leave those 
implementations (which are generally straightforward) to exercises and booksite code. 
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Tries In this section, we consider a search tree known as a trie , a data structure built 
from the characters of the string keys that allows us to use the characters of the search 
key to guide the search. The name “trie” is a bit of wordplay introduced by E. Fredkin in 
1960 because the data structure is used for retrieval, but we pronounce it “try” to avoid 
confusion with “tree.” We begin with a high-level description of the basic properties 
of tries, including search and insert algorithms, and then proceed to the details of the 
representation and lava implementation. 

Basic properties. As with search trees, tries are data structures composed of nodes that 
contain links that are either null or references to other nodes. Each node is pointed to 
by just one other node, which is called its parent (except for one node, the root, which 
has no nodes pointing to it), and each node has R links, where R is the alphabet size. 
Often, tries have a substantial number of null links, so when we draw a trie, we typi¬ 
cally omit null links. Although links point to nodes, we can view each link as point¬ 
ing to a trie, the trie whose root is the referenced node. Each link corresponds to a 
character value—since each link points 
to exactly one node, we label each node 
with the character value corresponding 
to the link that points to it (except for 
the root, which has no link pointing to 
it). Each node also has a corresponding 
value, which may be null or the value as¬ 
sociated with one of the string keys in 
the symbol table. Specifically, we store 
the value associated with each key in the 
node corresponding to its last character. 

It is very important to bear in mind the 
following fact: nodes with null values ex¬ 
ist to facilitate search in the trie and do 
not correspond to keys. An example of a 
trie is shown at right. 

Search in a trie. Finding the value as¬ 
sociated with a given string key in a trie 
in the search key. Each node in the trie has a link corresponding to each possible string 
character. We start at the root, then follow the link associated with the first character in 
the key; from that node we follow the link associated with the second character in the 
key; from that node we follow the link associated with the third character in the key and 



link to trie for all keys 
that start with s 


link to trie for all keys 
that start with she 

value for she in node 
corresponding to 
last key character 


label each node with 
character associated 
with incoming link 


key value 


Anatomy of a trie 


is a simple process, guided by the characters 
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hits 


misses 




return the value in the the value in the node 

node corresponding to corresponding to the last key 

the last key character character is n u T1, so return null 



so forth, until reaching the last character of the key or a null link. At this point, one of 
the following three conditions holds (refer to the figure above for examples): 

■ The value at the node corresponding to the last character in the key is not null 
(as in the searches for shel 1 s and she depicted at left above). This result is 

a search hit —the value associated with the key is the value in the node corre¬ 
sponding to its last character. 

■ The value in the node corresponding to the last character in the key is nul 1 (as 
in the search for shell depicted at top right above). This result is a search miss: 
the key is not in the table. 

■ The search terminated with a null link (as in the search for shore depicted at 
bottom right above). This result is also a search miss. 

In all cases, the search is accomplished just by examining nodes along a path from the 
root to another node in the trie. 
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Insertion into a trie. As with binary search trees, we insert by first doing a search: in 
a trie that means using the characters of the key to guide us down the trie until reach¬ 
ing the last character of the key or a null link. At this point, one of the following two 
conditions holds: 

■ We encountered a null link before reaching the last character of the key. In this 
case, there is no trie node corresponding to the last character in the key, so we 
need to create nodes for each of the characters in the key not yet encountered 
and set the value in the last one to the value to be associated with the key. 

■ We encountered the last character of the key before reaching a null link. In this 
case, we set that node’s value to the value to be associated with the key (whether 
or not that value is null), as usual with our associative array convention. 

In all cases, we examine or create a node in the trie for each key character. The construc¬ 
tion of the trie for our standard indexing client from chapter 3 with the input 

she sells sea shells by the sea shore 

is shown on the facing page. 

Node representation. As mentioned at the outset, our trie diagrams do not quite cor¬ 
respond to the data structures our programs will build, because we do not draw null 
links. Taking null links into account emphasizes the following important characteristics 
of tries: 

■ Every node has R links, one for each possible character. 

■ Characters and keys are implicitly stored in the data structure. 

For example, the figure below depicts a trie for keys made up of lowercase letters, with 
each node having a value and 26 links. The first link points to a subtrie for keys begin¬ 
ning with a, the second points to a subtrie for substrings beginning with b, and so forth. 



Trie representation (R = 26) 
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key 

she 


value 


root 


value 



key 

by 


value is in node 
corresponding to 
(h) / last character 



key is sequence 
of characters from 
root to value 


shells 3 


nodes corresponding to 
characters at the end of the 
key do not exist, so create them 
and set the value of the last one 



the 


sea 6 


node corresponding to 
the last key character 
exists, so reset its value 


shore 7 


Trie construction trace for standard indexing client 
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Keys in the trie are implicitly represented by paths from the root that end at nodes with 
non-null values. For example, the string sea is associated with the value 2 in the trie 
because the 19th link in the root (which points to the trie for all keys that start with s) 
is not null and the 5th link in the node that link refers to (which points to the trie for 
all keys that start with se) is not null, and the first link in the node that link refers to 
(which points to the trie for all keys that starts with sea) has the value 2. Neither the 
string sea nor the characters s, e, and a are stored in the data structure. Indeed, the data 
structure contains no characters or strings, just links and values. Since the parameter R 
plays such a critical role, we refer to a trie for an /^-character alphabet as an R-way trie. 

with these preparations, the symbol-table implementation TrieST on the facing 
page is straightforward. It uses recursive methods like those that we used for search 
trees in chapter 3 , based on a private Node class with instance variable val for client 
values and an array next[] of Node references. The methods are compact recursive 
implementations that are worthy of careful study. Next, we discuss implementations of 
the constructor that takes an A1 phabet as argument and the methods si ze(), keys(), 
longestPrefixOfO, keysWithPrefix(), keysThatMatchO, and deleteO. These 
are also easily understood recursive methods, each slightly more complicated than the 
last. 

Size. As for the binary search trees of chapter 3 , three straightforward options are 
available for implementing si ze (): 

■ An eager implementation where we maintain the number of keys in an instance 
variable N. 

■ A very eager implementation where we maintain the number of keys in a subtrie 
as a node instance variable that we update after the recursive calls in put() and 
delete(). 

■ A lazy recursive implementation like 
the one at right. It traverses all of the 
nodes in the trie, counting the number 
having a non-null value. 

As with binary search trees, the lazy im¬ 
plementation is instructive but should be 
avoided because it can lead to performance 
problems for clients. The eager implementa¬ 
tions are explored in the exercises. 

Lazy recursive size() for tries 


public int size() 

{ return size(root); } 

private int size(Node x) 

{ 

if (x == null) return 0; 
int cnt = 0; 

if (x.val != null) cnt++; 
for (char c = 0; c < R; C++) 
cnt += size(next[c]); 

return cnt; 

} 
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ALGORITHM 5.4 Trie symbol table 


public class TrieST<Value> 

{ 

private static int R = 256; // radix 
private Node root; // root of trie 

private static class Node 

{ 

private Object val; 

private Node[] next = new Node[R]; 

} 

public Value get(String key) 

{ 

Node x = get(root, key, 0); 
if (x == null) return null; 
return (Value) x.val; 

} 

private Node get(Node x, String key, int d) 

{ // Return value associated with key in the subtrie rooted at x. 

if (x == null) return null; 
if (d == key.lengthO) return x; 

char c = key.charAt(d); // Use dth key char to identify subtrie. 
return get(x.next[c], key, d+1); 

} 

public void put(String key, Value val) 

{ root = put(root, key, val, 0); } 

private Node put(Node x, String key, Value val, int d) 

{ // Change value associated with key if in subtrie rooted at x. 

if (x == null) x = new Node(); 

if (d == key.lengthO) { x.val = val; return x; } 

char c = key.charAt(d); // Use dth key char to identify subtrie. 

x.next[c] = put(x.next[c], key, val, d+1); 
return x; 

} 

} 


This code uses an R -way trie to implement a symbol table. Additional methods in the string symbol- 
table API of page 730 are presented in the next several pages. Modifying this code to handle keys from 
specialized alphabets is straighforward (see page 740). The value in Node has to be an Object because 
Java does not support arrays of generics; we cast values back to Val ue in get(). 
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Collecting keys. Because characters and keys are represented implicitly in tries, provid¬ 
ing clients with the ability to iterate through the keys presents a challenge. As with bina¬ 
ry search trees, we accumulate the string keys in a Queue, but for tries we need to create 
explicit representations of all of the string keys, not just find them in the data structure. 
We do so with a recursive private method col 1 ect () that is similar to si ze () but also 
maintains a string with the sequence of characters on the path from the root. Each time 
that we visit a node via a call to col 1 ect O with that node as first argument, the second 

argument is the string associated 
with that node (the sequence of 

public Iterable<String> keys() 

{ return keysWi thPrefi x("") ; } characters on the path from the 


public Iterable<String> keysWithPrefix(Str 
{ 

Queue<String> q = new Queue<String>(); 
collect(get(root, pre, 0), pre, q); 
return q; 

} 

private void collect(Node x, String pre, 

Queue<String> 

{ 

if (x == null) return; 

if (x.val != null) q.enqueue(pre); 

for (char c = 0; c < R; C++) 

collect(x.next[c], pre + c, q); 

} 


Collecting the keys in a trie 



sho 

shor 

shore by sea sells she shells shore 


t 

th 

the by sea sells she shells shore the 

Collecting the keys in a trie (trace) 


. root to the node). To visit a node, 
ng pre) 

we add its associated string to the 
queue if its value is not null, then 
visit (recursively) all the nodes 
in its array of links, one for each 
possible character. To create the 
q) key for each call, we append the 

character corresponding to the 
link to the current key. We use 
this collectO method to col¬ 
lect keys for both the keys() and 
the keysWi thPrefi x() methods 
in the API. To implement keys () 
we call keysWi thPrefi x() with 
the empty string as argument; to implement 
keysWi thPrefixO, we call get() to find the 
trie node corresponding to the given prefix 
(null if there is no such node), then use the 
collectO method to complete the job. The 
diagram at left shows a trace of collectO 
(or keysWi thPrefi x("")) for an example 
trie, giving the value of the second argument 
key and the contents of the queue for each 
call to collectO. The diagram at the top 
of the facing page illustrates the process for 
keysWithPrefix("sh"). 
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Wildcard match. To implement keysThatMatchO, we use a similar process, but add 
an argument specifying the pattern to col 1 ect () and add a test to make a recursive call 
for all links when the pattern character is a wildcard or only for the link corresponding 
to the pattern character otherwise, as in the code below. Note also that we do not need 
to consider keys longer than the pattern. 

Longest prefix. To find the longest key that is a prefix of a given string, we use a recur¬ 
sive method like get() that keeps track of the length of the longest key found on the 
search path (by passing it as a parameter to the recursive method, updating the value 


public Iterable<String> keysThatMatch(String pat) 

{ 

Queue<String> q = new Queue<String>(); 
collect(root, pat, q); 
return q; 

} 

public void collect(Node x, String pre, String pat, Queue<String> q) 
{ 

int d = pre.lengthO; 
if (x == null) return; 

if (d == pat.length() && x.val != null) q.enqueue(pre); 
if (d == pat.length()) return; 

char next = pat.charAt(d); 
for (char c = 0; c < R; C++) 
if (next == 1 . 1 || next == c) 

collect(x.next[c], pre + c, pat, q); 

} 

Wildcard match in a trie 
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public String longestPrefixOf(String s) 

{ 

int length = search(root, s, 0, 0); 
return s.substring(0, length); 

} 

private int search(Node x, String s, int d, int length) 
{ 

if (x == null) return length; 
if (x.val != null) length = d; 
if (d == s.lengthO) return length; 
char c = s.charAt(d); 

return search(x.next[c], s, d+1, length); 

} 

Matching the longest prefix of a given string 


whenever a node with a non-null value is encountered). The 
search ends when the end of the string or a null link is en¬ 
countered, whichever comes first. 

Deletion. The first step needed to delete a key-value pair 
from a trie is to use a normal search to find the node cor¬ 
responding to the key and set the corresponding value to 
null. If that node has a non-null link to a child, then no 
more work is required; if all the links are null, we need to 
remove the node from the data structure. If doing so leaves 
all the links null in its parent, we need to remove that node, 
and so forth. The implementation on the facing page dem¬ 
onstrates that this action can be accomplished with remark¬ 
ably little code, using our standard recursive setup: after the 
recursive calls for a node x, we return null if the client value 
and all of the links in a node are null; otherwise we return 
x. 




(last key on path) 

"shelters" 



Possibilities for 1 ongestPrefi xOf () 
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Alphabet. As usual, algorithm 5.4 
is coded for Java Stri ng keys, but it is 
a simple matter to modify the imple¬ 
mentation to handle keys taken from 
any alphabet, as follows: 

■ Implement a constructor that 
takes an A1 phabet as argument, 
which sets an A1 phabet in¬ 
stance variable to that argument 
value and the instance variable 
R to the number of characters 
in the alphabet. 

■ Use the toIndexO method 
from A1 phabet in get() and 
put() to convert string char¬ 
acters to indices between 0 and 
R-l. 

■ Use the toChar () method from 
A1 phabet to convert indices between 0 and R— 1 to char values. This operation 
is not needed in get() and put() but is important in the implementations of 
keysO, keysWithPrefixO, and keysThatMatchC). 

With these changes, you can save a considerable amount of space (use only R links per 
node) when you know that your keys are taken from a small alphabet, at the cost of the 
time required to do the conversions between characters and indices. 


public void delete(String key) 

{ root = delete(root, key, 0); } 

private Node delete(Node x, String key, int d) 
{ 

if (x == null) return null; 
if (d == key.lengthO) 
x.val = null; 
el se 
{ 

char c = key.charAt(d); 

x.nextfc] = delete(x.next[c], key, d+1); 

} 

if (x.val != null) return x; 

for (char c = 0; c < R; C++) 

if (x.nextjc] != null) return x; 
return nul1; 


Deleting a key (and its associated value) from a trie 


delete("shells"); 



Deleting a key (and its associated value) from a trie 
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the code that we have considered is a compact and complete implementation 
of the string symbol-table API that has broadly useful practical applications. Several 
variations and extensions are discussed in the exercises. Next, we consider basic proper¬ 
ties of tries, and some limitations on their utility. 

Properties of tries As usual, we are interested in knowing the amount of time and 
space required to use tries in typical applications. Tries have been extensively studied 
and analyzed, and their basic properties are relatively easy to understand and to apply. 

Proposition F. The linked structure (shape) of a trie is independent of the key in¬ 
sertion/deletion order: there is a unique trie for any given set of keys. 

Proof: Immediate, by induction on the subtries. 


This fundamental fact is a distinctive feature of tries: for all of the other search tree 
structures that we have considered so far, the tree that we construct depends both on 
the set of keys and on the order in which we insert those keys. 

Worst-case time bound for search and insert. How long does it take to find the value 
associated with a key? For BSTs, hashing, and other methods in chapter 4 , we needed 
mathematical analysis to study this question, but for tries it is very easy to answer: 

Proposition G. The number of array accesses when searching in a trie or inserting 
a key into a trie is at most 1 plus the length of the key. 

Proof: Immediate from the code. The recursive get() and put() implementations 
carry an argument d that starts at 0 , increments for each call, and is used to stop the 
recursion when it reaches the key length. 


From a theoretical standpoint, the implication of proposition g is that tries are opti¬ 
mal for search hit—we could not expect to do better than search time proportional to 
the length of the search key. Whatever algorithm or data structure we are using, we can¬ 
not know that we have found a key that we seek without examining all of its characters. 
From a practical standpoint this guarantee is important because it does not depend on 
the number of keys: when we are working with 7-character keys like license plate num¬ 
bers, we know that we need to examine at most 8 nodes to search or insert; when we are 
working with 20 -digit account numbers, we only need to examine at most 21 nodes to 
search or insert. 
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Expected time bound for search miss. Suppose that we are searching for a key in a 
trie and find that the link in the root node that corresponds to its first character is null. 
In this case, we know that the key is not in the table on the basis of examining just one 
node. This case is typical: one of the most important properties of tries is that search 
misses typically require examining just a few nodes. If we assume that the keys are 
drawn from the random string model (each character is equally likely to have any one 
of the R different character values) we can prove this fact: 

Proposition H. The average number of nodes examined for search miss in a trie 
built from N random keys over an alphabet of size R is ~log R N. 

Proof sketch (for readers who are familiar with probabilistic analysis): The prob¬ 
ability that each of the AT keys in a random trie differs from a random search key in 
at least one of the leading t characters is (1 — R~ f ) N . Subtracting this quantity from 
1 gives the probability that one of the keys in the trie matches the search key in all 
of the leading t characters. In other words, 1 — (1 — R~ f ) v is the probability that 
the search requires more than t character compares. From probabilistic analysis, 
the sum for t = 0 , 1 , 2 , ... of the probabilities that an integer random variable is >t 
is the average value of that random variable, so the average search cost is 

1 - (1 - R~ i) N + 1 - (1 - R~ 2 ) N + ...+ 1 - (1 - R~ f ) N +... 

Using the elementary approximation (1 — 1 lx) x ~ e~ l , we find the search cost to be 
approximately 

(1 -£-»«') + (1 - e -w« 2 ) +...+ (l - e ~ N ' Rt ) +... 

The summand is extremely close to 1 for approximately In R N terms with IV sub¬ 
stantially smaller than N; it is extremely close to 0 for all the terms with IV substan¬ 
tially greater than N; and it is somewhere between 0 and 1 for the few terms with 
R* ~ N. So the grand total is about log R N. 


From a practical standpoint, the most important implication of this proposition is that 
search miss does not depend on the key length. For example, it says that unsuccessful 
search in a trie built with 1 million random keys will require examining only three or 
four nodes, whether the keys are 7-digit license plates or 20-digit account numbers. 
While it is unreasonable to expect truly random keys in practical applications, it is rea¬ 
sonable to hypothesize that the behavior of trie algorithms for keys in typical applica- 
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tions is described by this model. Indeed, this sort of behavior is widely seen in practice 
and is an important reason for the widespread use of tries. 

Space. How much space is needed for a trie? Addressing this question (and under¬ 
standing how much space is available) is critical to using tries effectively. 

Proposition I. The number of links in a trie is between RN and RNw, where w is 
the average key length. 

Proof: Every key in the trie has a node containing its associated value that also has 
R links, so the number of links is at least RN. If the first characters of all the keys are 
different, then there is a node with R links for every key character, so the number of 
links is R times the total number of key characters, or RNw. 


put("shells", 1); 
put("shellfish", 2); 

standard 

trie 


no one-way 
branching 


The table on the facing page shows the costs for some typical applications that we have 
considered. It illustrates the following rules of thumb for tries: 

■ When keys are short, the number of links is close to RN. 

■ When keys are long, the number of links 
is close to RNw. 

■ Therefore, decreasing R can save a huge 
amount of space. 

A more subtle message of this table is that it is 
important to understand the properties of the 
keys to be inserted before deploying tries in an 
application. 

One-way branching. The primary reason 
that trie space is excessive for long keys is that 
long keys tend to have long tails in the trie, 
with each node having a single link to the next 
node (and, therefore, R— 1 null links). This 
situation is not difficult to correct (see exer¬ 
cise 5.2.11). A trie might also have internal 
one-way branching. For example, two long 
keys may be equal except for their last charac¬ 
ter. This situation is a bit more difficult to ad¬ 
dress (see exercise 5.2.12). These changes can 
make trie space usage a less important factor 

Removing one-way branching in a trie 
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than for the straightforward implementation that we have considered, but they are not 
necessarily effective in practical applications. Next, we consider an alternative approach 
to reducing space usage for tries. 

the bottom line is this: do not try to use algorithm 5.4/or large numbers of long keys 
taken from large alphabets, because it will require space proportional to R times the total 
number of key characters. Otherwise, if you can afford the space, trie performance is 
difficult to beat. 


application 

typical key 

average 

length 

w 

alphabet 

size 

R 

links in trie 
built from 1 million 

CA license plates 

4PGC938 

7 

256 

256 million 




256 

4 billion 

account numbers 

02400019992993299111 

20 






10 

256 million 

URLs 

www.es.princeton.edu 

28 

256 

4 billion 

text processing 

seashel1s 

11 

256 

256 million 

proteins in 

ACTGACTG 

8 

256 

256 million 

genomic data 



4 

4 million 


Space requirements for typical tries 
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get("sea") 


Ternary search tries (TSTs) To help us 

avoid the excessive space cost associated with 
R -way tries, we now consider an alternative 
representation: the ternary search trie (TST). 
In a TST, each node has a character, three links, 
and a value. The three links correspond to keys 
whose current characters are less than, equal 
to, or greater than the node’s character. In the 
R -way tries of algorithm 5.4, trie nodes are 
represented by R links, with the character cor¬ 
responding to each non-null link implictly 
represented by its index. In the corresponding 
TST, characters appear explicitly in nodes—we 
find characters corresponding to keys only 
when we are traversing the middle links. 

Search and insert. The search and insert code 
for implementing our symbol-table API with 
TSTs writes itself. To search, we compare the 
first character in the key with the character at 
the root. If it is less, we take the left link; if it is 
greater, we take the right link; and if it is equal, 
we take the middle link and move to the next 
search key character. In each case, we apply 

the algorithm 




match: take middle link, 
move to next char 


TST representation of a trie 



recursively. We 
terminate with 

a search miss if we encounter a null link or if the node 
where the search ends has a null value, and we terminate 
with a search hit if the node where the search ends has a 
non-null value. To insert a new key, we search, then add 
new nodes for the characters in the tail of the key, just as 
we did for tries, algorithm 5.5 gives the details of the 
implementation of these methods. 

Using this arrangement is equivalent to implement¬ 
ing each R -way trie node as a binary search tree that uses 
as keys the characters corresponding to non-null links. 
By contrast, algorithm 5.4 uses a key-indexed array. A 
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ALGORITHM 5.5 TST symbol table 


public class TST<Value> 

{ 

private Node root; 

private class Node 

{ 

char c; 

Node left, mid, right; 
Value val; 

} 


// root of trie 


// character 

// left, middle, and right subtries 
// value associated with string 


public Value get(String key) // same as for tries (See page 7B7). 

private Node get(Node x, String key, int d) 

{ 

if (x == null) return null; 
char c = key.charAt(d); 

if (c < x.c) return get(x.left, key, d); 

else if (c > x.c) return get(x.right, key, d); 
else if (d < key.lengthO - 1) 

return get(x.mid, key, d+1); 

else return x; 

} 


public void putfString key, Value val) 
{ root = put(root, key, val, 0); } 


private Node put(Node x, String key, Value val, int d) 

{ 

char c = key.charAt(d); 

if (x == null) { x = new Node(); x.c = c; } 
if (c < x.c) x.left = putfx.left, key, val, d); 

else if (c > x.c) x.right = put(x.right, key, val, d); 

else if (d < key.lengthO - 1) 

x.mid = putfx.mid, key, val, d+1); 

else x.val = val; 

return x; 

} 


This implementation uses a char value c and three links per node to build string search tries where 
subtries have keys whose first character is less than c (left), equal to c (middle), and greater than c 
(right). 
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TST and its corresponding trie are illustrated above. Continuing the correspondence 
described in chapter 3 between binary search trees and sorting algorithms, we see 
that TSTs correspond to 3-way string quicksort in the same way that BSTs correspond 
to quicksort and tries correspond to MSD sorting. The figures on page 714 and 721, 
which show the recursive call structure for MSD and 3-way three-way string quicksort 
(respectively), correspond precisely to the trie and TST drawn on page 746 for that set 
of keys. Space for links in tries corresponds to the space for counters in string sorting; 
3-way branching provides an effective solution to both problems. 


standard array of links (R = 26) 


ternary search tree (TST) 



link for keys - 
that start with su 


Trie node representations 
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Properties of TSTs A TST is a compact representation of an R -way trie, but the 
two data structures have remarkably different properties. Perhaps the most important 
difference is that property a does not hold for TSTs: the BST representations of each 
trie node depend on the order of key insertion, as with any other BST. 

Space. The most important property of TSTs is that they have just three links in each 
node, so a TST requires far less space than the corresponding trie. 

Proposition J. The number of links in a TST built from N string keys of average 
length w is between 3 N and 3 Nw. 

Proof. Immediate, by the same argument as for proposition i. 


Actual space usage is generally less than the upper bound of three links per character, 
because keys with common prefixes share nodes at high levels in the tree. 

Search cost. To determine the cost of search (and insert) in a TST, we multiply the cost 
for the corresponding trie by the cost of traversing the BST representation of each trie 
node.. 


Proposition K. A search miss in a TST built from N random string keys requires 
~ln N character compares, on the average. A search hit or an insertion in a TST uses 
a character compare for each character in the search key. 

Proof: The search hit/insertion cost is immediate from the code. The search miss 
cost is a consequence of the same arguments discussed in the proof sketch of prop¬ 
osition h. We assume that all but a constant number of the nodes on the search 
path (a few at the top) act as random BSTs on R character values with average path 
length In R, so we multiply the time cost log R N = In N /In R by In R. 


In the worst case, a node might be a full R -way node that is unbalanced, stretched out 
like a singly linked list, so we would need to multiply by a factor of R. More typically, 
we might expect to do In R or fewer character compares at the first level (since the root 
node behaves like a random BST on the R different character values) and perhaps at a 
few other levels (if there are keys with a common prefix and up to R different values on 
the character following the prefix), and to do only a few compares for most characters 
(since most trie nodes are sparsely populated with non-null links). Search misses are 
likely to involve only a few character compares, ending at a null link high in the trie, 
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and search hits involve only about one compare per search key character, since most of 
them are in nodes with one-way branching at the bottom of the trie. 

Alphabet. The prime virtue of using TSTs is that they adapt gracefully to irregularities 
in search keys that are likely to appear in practical applications. In particular, note that 
there is no reason to allow for strings to be built from a client-supplied alphabet, as was 
crucial for tries. There are two main effects. First, keys in practical applications come 
from large alphabets, and usage of particular characters in the character sets is far from 
uniform. With TSTs, we can use a 256-character ASCII encoding or a 65,536-character 
Unicode encoding without having to worry about the excessive costs of nodes with 256- 
or 65,536-way branching, and without having to determine which sets of characters are 
relevant. Unicode strings in non-Roman alphabets can have thousands of characters— 
TSTs are especially appropriate for standard Java Stri ng keys that consist of such char¬ 
acters. Second, keys in practical applications often have a structured format, differing 
from application to application, perhaps using only letters in one part of the key, only 
digits in another part of the key. In our CA license plate example, the second, third, and 
fourth characters are uppercase letter (R = 26) and the other characters are decimal 
digits (R = 10). In a TST for such keys, some of the trie nodes will be represented as 
10-node BSTs (for places where all keys have digits) and others will be represented as 
26-node BSTs (for places where all keys have letters). This structure develops automati¬ 
cally, without any need for special analysis of the keys. 

Prefix match, collecting keys, and wildcard match. Since a TST represents a 
trie, implementations of longestPrefixOfO, keys(), keysWithPrefixO, and 
keysThatMatchO are easily adapted from the corresponding code for tries in the pre¬ 
vious section, and a worthwhile exercise for you to cement your understanding of both 
tries and TSTs (see exercise 5 . 2 . 9 ). The same tradeoff as for search (linear memory 
usage but an extra In R multiplicative factor per character compare) holds. 

Deletion. The del ete() method for TSTs requires more work. Essentially, each char¬ 
acter in the key to be deleted belongs to a BST. In a trie, we could remove the link cor¬ 
responding to a character by setting the corresponding entry in the array of links to 
nul 1 ; in a TST, we have to use BST node deletion to remove the node corresponding to 
the character. 

Hybrid TSTs. An easy improvement to TST-based search is to use a large explicit mul¬ 
tiway node at the root. The simplest way to proceed is to keep a table of R TSTs: one 
for each possible value of the first character in the keys. If R is not large, we might use 
the first two letters of the keys (and a table of size R 2 ). For this method to be effec¬ 
tive, the leading digits of the keys must be well-distributed. The resulting hybrid search 
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algorithm corresponds to the way that a human might search for names in a telephone 
book. The first step is a multiway decision (“Let’s see, it starts with ‘A’,”), followed per¬ 
haps by some two-way decisions (“It’s before ‘Andrews,’ but after ‘Aitken,’”), followed 
by sequential character matching (“‘Algonquin,’... No, ‘Algorithms’ isn’t listed, because 
nothing starts with ‘Algor’!”). These programs are likely to be among the fastest avail¬ 
able for searching with string keys. 

One-way branching. Just as with tries, we can make TSTs more efficient in their use of 
space by putting keys in leaves at the point where they are distinguished and by elimi¬ 
nating one-way branching between internal nodes. 

Proposition L. A search or an insertion in a TST built from N random string keys 
with no external one-way branching and R f -way branching at the root requires 
roughly In N — tin R character compares, on the average. 

Proof: These rough estimates follow from the same argument we used to prove 
proposition k. We assume that all but a constant number of the nodes on the 
search path (a few at the top) act as random BSTs on R character values, so we 
multiply the time cost by In R. 


despite the temptation to tune the algorithm to peak performance, we should not 
lose sight of the fact that one of the most attractive features of TSTs is that they free us 
from having to worry about application-specific dependencies, often providing good 
performance without any tuning. 
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Which string symbol-table implementation should I use? As with string 
sorting, we are naturally interested in how the string-searching methods that we have 
considered compare to the general-purpose methods that we considered in chapter 
3 . The following table summarizes the important characteristics of the algorithms that 
we have discussed in this section (the rows for BSTs, red-black BSTs and hashing are 
included from chapter 3 , for comparison). For a particular application, these en¬ 
tries must be taken as indicative, not definitive, since so many factors (such as char¬ 
acteristics of keys and mix of operations) come into play when studying symbol-table 
implementations. 


typical growth rate for N strings 
from an R-character alphabet 
algorithm (average length w) 

(data structure) characters 

examined for memory usage 

search miss 


binary tree search 
(BST) 

c, (lg N) 2 

64 N 

randomly ordered 
keys 

2-3 tree search 
(red-black BST) 

c 2 (\gN) 2 

64 N 

guaranteed 

performance 

linear probing^ 
(parallel arrays) 

w 

32 JV to 128N 

built-in types 
cached hash values 

trie search 
(R-way trie) 

log rN 

(8R+56)N to (8R+56)Nw 

short keys 
small alphabets 

trie search 
(TST) 

1.39 lg N 

64 N to 64iVw 

nonrandom keys 


Performance characteristics of string-searching algorithms 


If space is available, R -way tries provide the fastest search, essentially completing the 
job with a constant number of character compares. For large alphabets, where space 
may not be available for R-wsy tries, TSTs are preferable, since they use a logarithmic 
number of character compares, while BSTs use a logarithmic number of key compares. 
Hashing can be competitive, but, as usual, cannot support ordered symbol-table opera¬ 
tions or extended character-based API operations such as prefix or wildcard match. 
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| Q&A 

Q. Does the Java system sort use one of these methods for searching with String keys? 
A. No. 


www.it-ebooks.info 





754 


CHAPTERS ■ Strings 


EXERCISES 

5 . 2.1 Draw the R -way trie that results when the keys 

no is th ti fo al go pe to co to th ai of th pa 
are inserted in that order into an initially empty trie (do not draw null links). 

5 . 2.2 Draw the TST that results when the keys 

no is th ti fo al go pe to co to th ai of th pa 
are inserted in that order into an initially empty TST. 

5 . 2.3 Draw the R -way trie that results when the keys 

now is the time for all good people to come to the aid of 
are inserted in that order into an initially empty trie (do not draw null links). 

5 . 2.4 Draw the TST that results when the keys 

now is the time for all good people to come to the aid of 
are inserted in that order into an initially empty TST. 

5 . 2.5 Develop nonrecursive versions of Tri eST and TST. 

5 . 2.6 Implement the following API, for a Stri ngSET data type: 


public class StringSET 



StringSET() 

create a string set 

voi d 

add(String key) 

put key into the set 

voi d 

delete(String key) 

remove key from the set 

boolean 

contains(String key) 

is key in the set? 

boolean 

i sEmptyO 

is the set empty? 

int 

size() 

number of keys in the set 

int 

toString() 

string representation of the set 


API for a string set data type 
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CREATIVE PROBLEMS 


5.2.7 Empty string in TSTs. The code in TST does not handle the empty string prop¬ 
erly. Explain the problem and suggest a correction. 

5.2.8 Ordered operations for tries. Implement the floorO, ceil(), rankO, and 
sel ect() (from our standard ordered ST API from chapter 3 ) for T ri eST. 

5.2.9 Extended operations for TSTs. Implement keys () and the extended opera¬ 
tions introduced in this section—longestPrefixOfO, keysWithPrefixO, and 
keysThatMatch()—for TST. 

5.2.10 Size. Implement very eager size() (that keeps in each node the number of 
keys in its subtree) for Tri eST and TST. 

5.2.11 External one-way branching. Add code to Tri eST and TST to eliminate external 
one-way branching. 

5.2.12 Internal one-way branching. Add code to Tri eST and TST to eliminate internal 
one-way branching. 

5.2.13 Hybrid TST with R 2 -way branching at the root. Add code to TST to do multiway 
branching at the first two levels, as described in the text. 

5.2.14 Unique substrings of length L. Write a TST client that reads in text from stan¬ 
dard input and calculates the number of unique substrings of length L that it contains. 
For example, if the input is cgcgggcgcg, then there are five unique substrings of length 
3: cgc, egg, geg, ggc, and ggg. Hint: Use the string method substri ng(i, i + L) to 
extract the i th substring, then insert it into a symbol table. 

5.2.15 Unique substrings. Write a TST client that reads in text from standard input 
and calculates the number of distinct substrings of any length. This can be done very 
efficiently with a suffix tree—see chapter 6 

5.2.16 Document similarity. Write a TST client with a static method that takes an i nt 
value L and two file names as command-line arguments and computes the L-similarity 
of the two documents: the Euclidean distance between the frequency vectors defined by 
the number of occurrences of each trigram divided by the number of trigrams. Include 
a static method mai n() that takes an i nt value L as command-line argument and a list 
of file names from standard input and prints a matrix showing the L-similarity of all 
pairs of documents. 
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CREATIVE PROBLEMS (continued) 


5.2.17 Spell checking. Write a TST client Spel 1 Checker that takes as command-line 
argument the name of a file containing a dictionary of words in the English language, 
and then reads a string from standard input and prints out any word that is not in the 
dictionary. Use a string set. 

5.2.18 Whitelist. Write a TST client that solves the whitelisting problem presented in 
section 1.1 and revisited in section 3.5 (see page 491). 

5.2.19 Random phone numbers. Write a TrieST client (with R = 10) that takes as 
command line argument an i nt value N and prints N random phone numbers of the 
form (xxx) xxx-xxxx. Use a symbol table to avoid choosing the same number more 
than once. Use the file AreaCodes. txt from the booksite to avoid printing out bogus 
area codes. 

5.2.20 Contains prefix. Add a method contai nsPrefi x() to StringSET (see exer¬ 
cise 5 . 2 . 6 ) that takes a string s as input and returns true if there is a string in the set 
that contains s as a prefix. 

5.2.21 Substring matches. Given a list of (short) strings, your goal is to support que¬ 
ries where the user looks up a string s and your job is to report back all strings in the 
list that contain s. Design an API for this task and develop a TST client that implements 
your API. Hint: Insert the suffixes of each word (e.g., stri ng, tri ng, ri ng, i ng, ng, g) 
into the TST. 

5.2.22 Typing monkeys. Suppose that a typing monkey creates random words by ap¬ 
pending each of 26 possible letter with probability p to the current word and finishes the 
word with probability 1 — 26 p. Write a program to estimate the frequency distribution 
of the lengths of words produced. If "abc" is produced more than once, count it only 
once. 
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| EXPERIMENTS 


5 . 2.23 Duplicates (revisited again). Redo exercise 3 . 5.30 using StringSET (see ex¬ 
ercise 5 . 2 . 6 ) instead of HashSET. Compare the running times of the two approaches. 
Then use Dedup to run the experiments for N = 10 7 , 10 8 , andlO 9 , repeat the experi¬ 
ments for random 1 ong values and discuss the results. 

5 . 2.24 Spell checker. Redo exercise 3 . 5 . 31 , which uses the file di cti onary.txt from 
thebooksite and the B1 ackFi 1 ter client on page 491 to print all misspelled words in a text 
file. Compare the performance of Tri eST and TST for the file war. txt with this client 
and discuss the results. 

5 . 2.25 Dictionary. Redo exercise 3 . 5 . 32 : Study the performance of a client like 
LookupCSV (using Tri eST and TST) in a scenario where performance matters. Specifi¬ 
cally, design a query-generation scenario instead of taking commands from standard 
input, and run performance tests for large inputs and large numbers of queries. 

5 . 2.26 Indexing. Redo exercise 3 . 5 . 33 : Study a client like Lookuplndex (using 
Tri eST and TST) in a scenario where performance matters. Specifically, design a query- 
generation scenario instead of taking commands from standard input, and run perfor¬ 
mance tests for large inputs and large numbers of queries. 
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A fundamental operation on strings is substring search: given a text string of length 
N and a pattern string of length M, find an occurrence of the pattern within the text. 
Most algorithms for this problem can easily be extended to find all occurrences of the 
pattern in the text, to count the number of occurrences of the pattern in the text, or to 
provide context (substrings of the text surrounding each occurrence of the pattern). 

When you search for a word while using a text editor or a web browser, you are doing 
substring search. Indeed, the original motivation for this problem was to support such 
searches. Another classic application is searching for some important pattern in an in¬ 
tercepted communication. A military leader might be interested in finding the pattern 
ATTACK AT DAWN somewhere in an intercepted text message; a hacker might be 
interested in finding the pattern Password : somewhere in your computer’s mem¬ 
ory. In today’s world, we are often searching through the vast amount of information 
available on the web. 

To best appreciate the algorithms, think of the pattern as being relatively short (with 
M equal to, say, 100 or 1,000) and the text as being relatively long (with N equal to, say, 
1 million or 1 billion). In substring search, we typically preprocess the pattern in order 
to be able to support fast searches for that pattern in the text. 

Substring search is an interesting and classic problem: several very different (and 
surprising) algorithms have been discovered that not only provide a spectrum of use¬ 
ful practical methods but also illustrate a spectrum of fundamental algorithm design 
techniques. 


pattern —► NEEDLE 

text —*- INAHAYSTACKNEEDLEINA 


match 
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A short history The algorithms that we examine have an interesting history; we 
summarize it here to help place the various methods io perspective. 

There is a simple brute-force algorithm for substring search that is in widespread 
use. While it has a worst-case running time proportional to MN, the strings that arise 
in many applications lead to a running time that is (except in pathological cases) pro¬ 
portional to M + N. Furthermore, it is well-suited to standard architectural features on 
most computer systems, so an optimized version provides a standard benchmark that 
is difficult to beat, even with a clever algorithm. 

In 1970, S. Cook proved a theoretical result about a particular type of abstract ma¬ 
chine that implied the existence of an algorithm that solves the substring search prob¬ 
lem in time proportional to M + Nin the worst case. D. E. Knuth and V. R. Pratt labori¬ 
ously followed through the construction Cook used to prove his theorem (which was 
not intended to be practical) and refined it into a relatively simple and practical algo¬ 
rithm. This seemed a rare and satisfying example of a theoretical result with immediate 
(and unexpected) practical applicability. But it turned out that J. H. Morris had discov¬ 
ered virtually the same algorithm as a solution to an annoying problem confronting 
him when implementing a text editor (he wanted to avoid having to “back up” in the 
text string). The fact that the same algorithm arose from two such different approaches 
lends it credibility as a fundamental solution to the problem. 

Knuth, Morris, and Pratt didn’t get around to publishing their algorithm until 1976, 
and in the meantime R. S. Boyer and J. S. Moore (and, independently, R. W. Gosper) 
discovered an algorithm that is much faster in many applications, since it often exam¬ 
ines only a fraction of the characters in the text string. Many text editors use this algo¬ 
rithm to achieve a noticeable decrease in response time for substring search. 

Both the Knuth-Morris-Pratt (KMP) and the Boyer-Moore algorithms require some 
complicated preprocessing on the pattern that is difficult to understand and has lim¬ 
ited the extent to which they are used. (In fact, the story goes that an unknown systems 
programmer found Morris’s algorithm too difficult to understand and replaced it with 
a brute-force implementation.) 

In 1980, M. O. Rabin and R. M. Karp used hashing to develop an algorithm almost as 
simple as the brute-force algorithm that runs in time proportional to M + N with very 
high probability. Furthermore, their algorithm extends to two-dimensional patterns 
and text, which makes it more useful than the others for image processing. 

This story illustrates that the search for a better algorithm is still very often justified; 
indeed, one suspects that there are still more developments on the horizon even for this 
classic problem. 
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Brute-force substring search An obvious method for substring search is to 
check, for each possible position in the text at which the pattern could match, whether 
it does in fact match. The search () method below operates in this way to find the first 
occurrence of a pattern string pat in a text string txt. The program keeps one pointer 

(i) into the text and another point¬ 
er (j) into the pattern. For each i, it 
resets j to 0 and increments it until 
finding a mismatch or the end of 
the pattern (j == M). If we reach 
the end of the text (i == N-M+l) 
before the end of the pattern, then 
there is no match: the pattern does 
not occur in the text. Our conven¬ 
tion is to return the value N to indi¬ 
cate a mismatch. 

In a typical text-processing ap¬ 
plication, the j index rarely incre¬ 
ments so the running time is pro¬ 
portional to N. Nearly all of the 
compares find a mismatch with the first character of the pattern. For example, suppose 
that you search for the pattern pattern in the text of this paragraph. There are 191 
characters up to the end of the first occurrence of the pattern, only 7 of which are the 
character p (and there are no occurrences of pa), so the total number of character com¬ 
pares is 191+7, for an average of 1.036 compares per character in the text. On the other 
hand, there is no guarantee that the algorithm will always be so efficient. For example, a 
pattern might begin with a long string of As. If it does, and the text also has long strings 
of As, then the substring search 

will be slow. i j i+j 0 1 2 3 4 5 6 7 8 9 10 

txt - 

0 2 2 

10 1 

2 1 3 

3 0 3 

4 1 5 

5 0 5 

6 4 10 

x return i when j is M \ 

match 


entries in black 
match the text 



B R A C 


entries in red are 
mismatches 


entries in gray are 
for reference only 

A 



public static int search(String pat, String txt) 
i 

int M = pat.lengthO; 
int N = txt.lengthO; 
for (int i =0; i <= N - M; i++) 

{ 

int j ; 

for (j = 0; j < M; j++) 

if (txt.charAt(i+j) != pat.charAt(j)) 
break; 

if (j == M) return i; // found 

} 

return N; // not found 


Brute-force substring search 


Brute-force substring search 
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Proposition M. Brute-force substring search requires ~NM character compares to 
search for a pattern of length M in a text of length N, in the worst case. 

Proof: A worst-case input is when both pattern and text are all As followed by a B. 
Then for each of the N— M + 1 possible match positions, all the characters in the 
pattern are checked against the text, for a total cost of M{N~ M + 1). Normally M 
is very small compared to N, so the total is ~NM. 


1 i+J 


0123456789 


txt - 

4 

5 

6 

7 

8 

10 


A A B 


Such degenerate strings are not likely to appear in English text, but they may well occur 
in other applications (for example, in binary texts), so we seek better algorithms. 

The alternate implementation at 
the bottom of this page is instructive. 
As before, the program keeps one 
pointer (i) into the text and another 
pointer (j) into the pattern. As long 
as they point to matching characters, 
both pointers are incremented. This 
code performs precisely the same 
character compares as the previous 
implementation. To understand it, 
note that i in this code maintains the value of i + j in the previous code: it points to the 
end of the sequence of already-matched characters in the text (where i pointed to the 
beginning of the sequence before). If i and j point to mismatching characters, then we 
back up both pointers: j to point to the beginning of the pattern and i to correspond to 
moving the pattern to the right one position for matching against the text. 


A 

pat 

B 

A 

A 

A 


Brute-force substring search (worst case) 


public static int search(String pat, String txt) 
{ 

int j, M = pat.length(); 
int i, N = txt.length(); 
for (i =0, j =0; i < N && j < M; i++) 

{ 

if (txt.charAt(i) == pat.charAt(j)) j++; 
else { i -= j; j = 0; } 

} 

if (j == M) return i - M; // found 
else return N; // not found 


Alternate implementation of brute-force substring search (explicit backup) 


www.it-ebooks.info 





762 


CHAPTER 5 ■ Strings 


Knuth-Morris-Pratt substring search The basic idea behind the algorithm 
discovered by Knuth, Morris, and Pratt is this: whenever we detect a mismatch, we 
already know some of the characters in the text (since they matched the pattern charac¬ 
ters prior to the mismatch). We can take advantage of this information to avoid backing 
up the text pointer over all those known characters. 

As a specific example, suppose that we have a two-character alphabet and are search¬ 
ing for the pattern BAAAAAAAAA. Now, suppose that we match five char¬ 
acters in the pattern, with a mismatch on the sixth. When the mismatch is detected, 

we know that the six previous 
characters in the text must 
be B A A A A B (the first 
five match and the sixth does 
not), with the text pointer 
now pointing at the B at the 
end. The key observation is 
that we need not back up the 
text pointer i, since the previ¬ 
ous four characters in the text 
are all As and do not match 
the first character in the pat¬ 
tern. Furthermore, the char¬ 
acter currently pointed to by 
i is a B and does match the first character in the pattern, so we can increment i and 
compare the next character in the text with the second character in the pattern. This 
argument leads to the observation that, for this pattern, we can change the else clause 
in the alternate brute-force implementation to just set j = 1 (and not decrement i). 
Since the value of i does not change within the loop, this method does at most N char¬ 
acter compares. The practical effect of this particular change is limited to this particular 
pattern, but the idea is worth thinking about—the Knuth-Morris-Pratt algorithm is a 
generalization of it. Surprisingly, it is always possible to find a value to set the j pointer 
to on a mismatch, so that the i pointer is never decremented. 

Fully skipping past all the matched characters when detecting a mismatch will not 
work when the pattern could match itself at any position overlapping the point of the 
mismatch. For example, when searching for the pattern A A B A A A in the text 
AABAABAAAA, we first detect the mismatch at position 5, but we had better 
restart at position 3 to continue the search, since otherwise we would miss the match. 
The insight of the KMP algorithm is that we can decide ahead of time exactly how to 
restart the search, because that decision depends only on the pattern. 




rexr-v 

after mismatch 
on sixth char 


brute-force backs 
up to try this 


and this ^ 

and this ^ 
and this ^ 
and this 


but no backup 
is needed 


I 

B 

A 

A 

A 

A 

A 

,B 


A A A A A A 

-— pattern 

A A 


BAAAAAAAAA 


Text pointer backup in substring searching 
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Backing up the pattern pointer. In KMP sub¬ 
string search, we never back up the text pointer 
i, and we use an array dfa[] [] to record how 
far to back up the pattern pointer j when a 
mismatch is detected. For every character c, 
dfa[c] [j] is the pattern position to compare 
against the next text position after compar¬ 
ing c with pat. charAt(j). During the search, 
dfa[txt.charAt(i)] [j] is the pattern position 
to compare with txt.charAt(i+l) after com¬ 
paring txt.charAt(i ) with pat.charAt(j). 
For a match, we want to just move on to the 
next character, so dfa[pat.charAt(j)] [j] is 
always j+1. For a mismatch, we know not just 
txt. charAt(i),but also the j-1 previous char¬ 
acters in the text: they are the first j-1 characters 
in the pattern. For each character c, imagine that 
we slide a copy of the pattern over these j char¬ 
acters (the first j-1 characters in the pattern fol¬ 
lowed by c —we are deciding what to do when 
these characters are txt.charAt(i-j+l. .i)), 
from left to right, stopping when all overlap¬ 
ping characters match (or there are none). 
This gives the next possible place the pattern 
could match. The index of the pattern char¬ 
acter to compare with txt.charAt(i+l) 
(dfa[txt.charAt(i)] [j]) is precisely the 
number of overlapping characters. 


j pat. charAt(j) dfa[] [j] text (pattern itself) 

ABC ABABAC 


0 A 


1 B 


2 A 


3 B 


1 A 

B 

0 ABABAC 

C 

0 

2 AB 

AA 

1 ABABAC 

AC 

0 

3 ABA 

ABB 

0 ABABAC 

ABC 

0 

4 ABAB 

ABAA 

1 ABABAC 

ABAC 

0 ABABAC 


4 A 


ABABA 

ABABB 

0 ABABAC 


match (move to next char) 
set dfa[pat.charAt(j)][j] 0 
to j+1 

5 C 6 


mismatch 
(back up in pattern) 


ABABC 

ABABAC 

known text char 
ABABAC / on mismatch 

ABABAA^^ 

ABABAC 

ABABAB 

ABABAC 


KMP search method. Once we have computed backup is length of max overlap 

. of beginning of pattern 

the dfa[] [] array, we have the substring search with known text chars 

method at the top of the next page: when i and Pattern backup for A b a b a c in kmp substring search 

j point to mismatching characters (testing for a 

pattern match beginning at position i - j+1 in the text string), then the next possible 
position for a pattern match is beginning at position i -dfa [txt. charAt (i) ] [ j ]. But 
by construction, the first df a [txt. charAt (i)] [j] characters at that position match 
the first dfa [txt. charAt (i )] [j] characters of the pattern, so there is no need to back 
up the i pointer: we can simply set j to dfa[txt.charAtfi )] [j] and increment i, 
which is precisely what we do when i and j point to matching characters. 
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DFA simulation. A useful way to describe this process is in terms of a deterministic 
finite-state automaton (DFA). Indeed, as indicated by its name, our dfa[] [] array pre¬ 
cisely defines a DFA. The graphical DFA represention shown at the bottom of this page 

consists of states (indicated by circled 


public int search(String txt) 


{ 


// Simulate operation of DFA on txt. 

int i, j, N = txt.lengthO; 
for (i =0, j =0; i < N && j < M; i++) 
j = dfa[txt.charAt(i)] [j] ; 
if (j == M) return i - M; // found 
else return N; // not found 


KMP substring search (DFA simulation) 


internal representation 


pat.charAt(j) 


dfa[][j] 


numbers) and transitions (indicated by 
labeled lines). There is one state for each 
character in the pattern, each such state 
having one transition leaving it for each 
character in the alphabet. For the sub- 
string-matching DFAs that we are con¬ 
sidering, one of the transitions is a match 
transition (going from j to j +1 and labeled 
with pat.charAt(j)) and all the others 
are mismatch transition (going left). The states correspond to character compares, one 
for each value of the pattern index. The transitions correspond to changing the value 
of the pattern index. When examining the text character i when in the state labeled j, 
the machine does the following: “Take the transition to dfa [txt. charAt (i ) ] [ j ] and 
move to the next character (by incrementing i).” For a match transition, we move to 
the right one position because dfa [pat. charAt(j)] [j] is always j+1; for a mismatch 
transition we move to the left. The automaton reads the text characters one at a time, 
from left to right, moving to a new state each time it reads a character. We also include a 
halt state M that has no transitions. We start the machine at state 0: if the machine reach¬ 
es state M, then a substring of the text 
matching the pattern has been found 
(and we say that the DFA recognizes the 
pattern); if the machine reaches the 
end of the text before reaching state 
M, then we know the pattern does not 
appear as a substring of the text. Each 
pattern corresponds to an automaton 
(which is represented by the dfa[] [] 
array that gives the transitions). The 
KMP substring search () method is 
a Java program that simulates the op¬ 
eration of such an automaton. 

To get a feeling for the operation of 
a substring search DFA, consider two 
of the simplest things that it does. At 


A 

1 

0 

0 


A 
3 
0 
0 

mismatch 

transition 


A 

5 

0 

0 


• match 
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0 
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return i - M = 9 
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match : 



A 

B 

A 

B 

A 

c 







set j to dfa[txt .charAt(i)] [j] 





A 

B 

A 

B 

A 

c 



= dfa[pat.charAt(j)][j] 













= j+i 







Aj, 

r B 

A 

B 

A 
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B 

A 

B 

A 

c 










A 

B 

A 

B 

A 
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mismatch: 






A 

B 

A 

B 

A 

c 



set j to dfa[txt. 

charAt(i)][j] 

A 

B 

A 

B 

A 

c 



implies pattern shift to align 










pat. 

charAt(j) with 



A 

B 

A 

B 

A 

c 



txt. 

charAt(i+l) 




A 

B 

A 

B 

A 

c 



Trace of KMP substring search (DFA simulation) for A B A B A C 


the beginning of the process, when started in state 0 at the beginning of the text, it stays 
in state 0, scanning text characters, until it finds a text character that is equal to the first 
pattern character, when it moves to the next state and is off and running. At the end of 
the process, when it finds a match, it matches pattern characters with the right end of 
the text, incrementing the state until reaching state M. The trace at the top of this page 
gives a typical example of the operation of our example DFA. Each match moves the 
DFA to the next state (which is equivalent to incrementing the pattern index j); each 
mismatch moves the DFA to an earlier state (which is equivalent to setting the pattern 
index j to a smaller value). The text index i marches from left to right, one position at 
a time, while the pattern index j bounces around in the pattern as directed by the DFA. 

Constructing the DFA. Now that you understand the mechanism, we are ready to ad¬ 
dress the key question for the KMP algorithm: How do we compute the dfa[] [] array 
corresponding to a given pattern? Remarkably, the answer to this question lies in the 
DFA itself (!) using the ingenious (and rather tricky) construction that was developed 
by Knuth, Morris, and Pratt. When we have a mismatch at pat. char At (j ), our interest 
is in knowing in what state the DFA would be if we were to back up the text index and 
rescan the text characters that we just saw after shifting to the right one position. We do 
not want to actually do the backup, just restart the DFA as if we had done the backup. 
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The key observation is that the characters in the text that would need to be rescanned 
are precisely pat.charAt(l) through pat.charAt(j-l): we drop the first character 
to shift right one position and the last character because of the mismatch. These are 
pattern characters that we know, so we can figure out ahead of 
time, for each possible mismatch position, the state where we 
need to restart the DFA. The figure at left shows the possibilities 
for our example. Be sure that you understand this concept. 

What should the DFA do with the next character? Exactly 
what it would have done if we had backed up, except if it finds 
a match with pat.charAt(j), when it should go to state j+1. 
For example, to decide what the DFA should do when we have 
a mismatch at j = 5 for A B A B A C, we use the DFA to learn 
that a full backup would leave us in state 3 for B A B A, so we 
can copy dfa[] [3] to dfa[] [5], then set the entry for C to 6 
because pat. charAt(5) is C (a match). Since we only need to 
know how the DFA runs for j -1 characters when we are build¬ 
ing the j th state, we can always get the information that we need 
from the partially built DFA. 

The final crucial detail to the computation is to observe that maintaining the restart 
position X when working on column j ofdfa[] [] is easy because X < j so that we can use 
the partially built DFA to do the job—the next value of X is dfa [pat. charAt ( j ) ] [X]). 
Continuing our example from the previous paragraph, we would update the value of 
X to dfa[ ' C' ] [ 3 ] = 0 (but we do not use that value because the DFA construction is 
complete). 

The discussion above leads to the remarkably compact code below for constructing 
the DFA corresponding to a given pattern. For each j, it 

■ Copies dfa[] [X] to dfa[] [j] (for mismatch cases) 

■ Sets dfa[pat.charAt(j)] [j] to j +1 (for the match case) 

■ Updates X 

The diagram on the facing page traces 
this code for our example. To make sure 
that you understand it, work exercise 
5.3.2 and exercise 5.3.3. 


Constructing the DFA for KMP substring search 


dfa[pat.charAt(0)][0] = 1; 

for (int X = 0, j = 1; j < M; j++) 

{ // Compute dfa[][j]. 

for (int c = 0; c < R; C++) 
dfa[c] [j] = dfa[c] [X] ; 
dfa[pat.charAt(j)][j] = j+1; 

X = dfa[pat.charAt(j)][X]; 

} 


1 A 



DFA simulations to compute 
restart states for A B A B A C 


www.it-ebooks.info 






5.3 


■ Substring Search 


767 


j 

pat.charAt(j) 


dfa[][j] 


A 

B 

C 


0 _ 

A 

1 

0 

0 



j 


—0 


X 

I 



j 

0 

1 





pat.charAt(j) 

A 

B 






A 

1 

1 





dfa[] [j] 

B 

0 

2 






C 

0 

0 







X 















j 

0 

1 

2 




pat.charAt(j) 

A 

B 

A 





A 

1 

1 

3 




dfa[] [j] 

B 

0 

2 

0 





C 

0 

0 

0 







X 








1 






j 

0 

1 

2 

3 



pat.charAt(j) 

A 

B 

A 

B 




A 

1 

1 

3 

1 



dfa[] [j] 

B 

0 

2 

0 

4 




C 

0 

0 

0 

0 







X 








1 





j 

0 

1 

2 

3 

4 


pat.charAt(j) 

A 

B 

A 

B 

A 



A 

1 

1 

3 

1 

5 


dfa[] [j] 

B 

0 

2 

0 

4 

0 



C 

0 

0 

0 

0 

0 







X 








1 




j 

0 

1 

2 

3 

4 

5 

pat.charAt(j) 

A 

B 

A 

B 

A 

C 


A 

1 

1 

3 

1 

5 

1 

dfa[][j] 

B 

0 

2 

0 

4 

0 

4 


C 

0 

0 

0 

0 

0 

6 



copy dfa[] [X] fodfa[] [j] 
dfa[pat.charAt(j)][j] = j+1; 
X = dfa[pat.charAt(j)] [X]; 



j 


—© 



Constructing the DFA for KMP substring search for A B A B A C 
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ALGORITHM 5.6 Knuth-Morris-Pratt substring search 


public class KMP 

{ 

private String pat; 
private int[][] dfa; 


public KMP(String pat) 

{ // Build DFA from pattern. 

this.pat = pat; 
int M = pat.lengthO; 
int R = 256; 
dfa = new int[R][M]; 
dfa[pat.charAt(O)][0] = 1; 
for (int X = 0, j = 1; j < M; j++) 

{ // Compute dfa[][j], 

for (int c = 0; c < R; C++) 
dfa[c][j] = dfa[c][X]; 
dfa[pat.charAt(j)][j] = j+1; 

X = dfa[pat.charAt(j)][X]; 

} 

} 

public int search(String txt) 

{ // Simulate operation of DFA on txt. 

int i, j, N = txt. 1 ength() , M = pat.lengthO; 

for (i =0, j =0; i < N && j < M; i++) 

j = dfa[txt.charAt(i)][j] ; 

if (j == M) return i - M; // found (hit end of pattern) 

else return N; // not found (hit end of text) 


// Copy mismatch cases. 
// Set match case. 

// Update restart state. 


public static void main(String[] args) 

// See page 769. 


The constructor in this implementation of the Knuth-Morris-Pratt algorithm for substring search 
builds a DFA from a pattern string, to support a search () method that can find the pattern in a given 
text string. This program does the same job as the brute-force method, but it runs faster for patterns 
that are self-repetitive. 

% java KMP AACAA AABRAACADABRAACAADABRA 
text: AABRAACADABRAACAADABRA 

pattern: AACAA 


www.it-ebooks.info 






5.3 ■ Substring Search 


769 


algorithm 5.6 on the facing page implements the following API: 
public class KMP 

KMPCString pat) create a DFA that can search for pat 

int search(Stri ng txt) find index of pat in txt 

Substring search API 

You can see a typical test client at the bottom of this page. The constructor builds a DFA 
from a pattern that the search () method uses to search for the pattern in a given text. 

Proposition N. Knuth-Morris-Pratt substring search accesses no more than M + N 
characters to search for a pattern of length M in a text of length N. 

Proof. Immediate from the code: we access each pattern character once when com¬ 
puting dfa[] [] and each text character once (in the worst case) in searchQ. 


Another parameter comes into play: for an P-character alphabet, the total running time 
(and space) required to build the DFA is proportional to MR. It is possible to remove 
the factor of R by building a DFA where each state has a match transition and a mis¬ 
match transition (not transitions for each possible character), though the construction 
is somewhat more intricate. 

The linear-time worst-case guarantee provided by the KMP algorithm is a significant 
theoretical result. In practice, the speedup over the brute-force method is not often 
important because few applications involve searching for highly self-repetitive patterns 
in highly self-repetitive text. Still, the method 
has the practical advantage that it never backs 
up in the input. This property makes KMP 
substring search more convenient for use 
on an input stream of undetermined length 
(such as standard input) than algorithms re¬ 
quiring backup, which need some complicat¬ 
ed buffering in this situation. Ironically, when 
backup is easy, we can do significantly better 
than KMP. Next, we consider a method that 
generally leads to substantial performance 
gains precisely because it can back up in the 
text. 


public static void main(String[] args) 
{ 

String pat = args[0]; 

String txt = args[l]; 

KMP kmp = new KMP(pat); 
StdOut.println("text: " + txt); 

int offset = kmp.search(txt); 
StdOut.print("pattern: "); 
for (int i =0; i < offset; i++) 
StdOut.print(" "); 

StdOut.println(pat); 

} 

KMP substring search test client 
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Boyer-Moore substring search When backup in the text string is not a prob¬ 
lem, we can develop a significantly faster substring-searching method by scanning the 
pattern from right to left when trying to match it against the text. For example, when 
searching for the substring BAABBAA, if we find matches on the seventh and sixth 
characters but not on the fifth, then we can immediately slide the pattern seven posi¬ 
tions to the right, and check the 14th character in the text next, because our partial 
match found X A A where X is not B , which does not appear elsewhere in the pattern. In 
general, the pattern at the end might appear elsewhere, so we need an array of restart 
positions as for Knuth-Morris-Pratt. We will not explore this approach in further detail 
because it is quite similar to our implementation of the Knuth-Morris-Pratt method. 
Instead, we will consider another suggestion by Boyer and Moore that is typically even 
more effective in right-to-left pattern scanning. 

As with our implementation of KMP substring search, we decide what to do next on 
the basis of the character that caused the mismatch in the text as well as the pattern. The 
preprocessing step is to decide, for each possible character that could occur in the text, 
what we would do if that character were to cause the mismatch. The simplest realiza¬ 
tion of this idea leads immediately to an efficient and useful substring search method. 

Mismatched character heuristic. Consider the figure at the bottom of this page, which 
shows a search for the pattern NEEDLE in the text FINDINAHAYSTACKNEEDLE. 
Proceeding from right to left to match the pattern, we first compare the rightmost E in 
the pattern with the N (the character at position 5) in the text. Since N appears in the 
pattern, we slide the pattern five positions to the right to line up the N in the text with 
the (rightmost) N in the pattern. Then we compare the rightmost E in the pattern with 
the S (the character at position 10) in the text. This is also a mismatch, but S does not 
appear in the pattern, so we can slide the pattern six positions to the right.We match 
the rightmost E in the pattern against the E at position 16 in the text, then find a mis¬ 
match and discover the N at position 15 and slide to the right five positions, as at the 
beginning. Finally, we verify, moving from right to left starting at position 20, that the 
pattern is in the text. This method brings us to the match position at a cost of only four 
character compares (and six more to verify the match)! 

i j 0 1 2 B 4 5 6 7 8 9 10 11 12 IB 14 15 16 17 18 19 20 21 22 23 

text — FINDINAHAYSTACKNEEDLE 

0 5 NEEDLE -* —pattern 

5 5 NEEDLE 

11 4 NEEDLE 

15 0 NEEDLE 

\ 

return i = 15 

Mismatched character heuristic for right-to-left (Boyer-Moore) substring search 
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Starting point. To implement the mismatched 
character heuristic, we use an array ri ght [] that 
gives, for each character in the alphabet, the index 
of its rightmost occurrence in the pattern (or -1 
if the character is not in the pattern). This value 
tells us precisely how far to skip if that character 
appears in the text and causes a mismatch during 
the string search. To initialize the ri ght [] array, 
we set all entries to -1 and then, for j from 0 to 
M-l, set right [pat. charAt(j)] to j, as shown 
in the example at right for our example pattern 
NEEDLE. 




N 

E 

E 

D 

L 

E 


c 


0 

1 

2 

B 

4 

5 

right[c] 

A 

-1 

-1 

-1 

-1 

-1 

-1 

-1 

-1 

B 

-1 

-1 

-1 

-1 

-1 

-1 

-1 

-1 

C 

-1 

-1 

-1 

-1 

-1 

-1 

-1 

-1 

D 

-1 

-1 

-1 

-1 

B 

3 

3 

3 

E 

-1 

-1 

1 

2 

2 

2 

5 

5 









-1 

L 

-1 

-1 

-1 

-1 

-1 

4 

4 

4 

M 

-1 

-1 

-1 

-1 

-1 

-1 

-1 

-1 

N 

-1 

0 

0 

0 

0 

0 

0 

0 









-1 


Boyer-Moore skip table computation 


Substring search. With the right[] array pre¬ 
computed, the implementation in algorithm 5.7 is straightforward. We have an index 
i moving from left to right through the text and an index j moving from right to left 
through the pattern. The inner loop tests whether the pattern aligns with the text at 
position i. If txt. charAt(i+j) is equal to pat. charAt(j) for all j from M-l down to 
0, then there is a match. Otherwise, there is a character mismatch, and we have one of 
the following three cases: 

■ If the character causing the mismatch is 
not found in the pattern, we can slide the 
pattern j+1 positions to the right (incre¬ 
ment i by j+1). Anything less would align 
that character with some pattern character. 

Actually, this move aligns some known 
characters at the beginning of the pattern 
with known characters at the end of the 
pattern so that we could further increase 
i by precomputing a KMP-like table (see 
example at right). 

■ If the character c causing the mismatch is 
found in the pattern, we use the ri ght [] array to line up the pattern with the 
text so that character will match its rightmost occurrence in the pattern. To do 
so, we increment i by j minus ri ght [c] . Again, anything less would align that 
text character with a pattern character it could not match (one to the right of its 
rightmost occurrence). Again, there is a possibility that we could do better with a 
KMP-like table, as indicated in the top example in the figure on page 773. 


E E 


i+] 

* 

T L 
D L 

t 

j 


increment i by j+1 I 

. . . T L 


could do better with 
/ KMP-like table 


E D L E 

reset j to M-l t 
j 


Mismatched character heuristic (mismatch not in pattern) 
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ALGORITHM 5.7 Boyer-Moore substring search (mismatched character heuristic) 


public class BoyerMoore 

{ 

private int[] right; 
private String pat; 


BoyerMoore(String pat) 

{ // Compute skip table. 

this.pat = pat; 

int M = pat.lengthO; 

int R = 256; 

ri ght = new int[R]; 

for (int c = 0; c < R; C++) 

right[c] = -1; // 

for (int j = 0; j < M; j++) // 

right[pat.charAt(j)] = j; // 


-1 for chars not in pattern 
rightmost position for 
chars in pattern 


public int search(String txt) 

{ // Search for pattern in txt. 

int N = txt.lengthO; 
int M = pat.lengthO; 
int skip; 

for (int i =0; i <= N-M; i += skip) 

{ // Does the pattern match the text at position i ? 

skip = 0; 

for (int j = M-l; j >= 0; j--) 

if (pat.charAt(j) != txt.charAt(i+j)) 

{ 

skip = j - right[txt.charAt(i+j)]; 

if (skip < 1) skip = 1; 

break; 

} 

if (skip == 0) return i; // found. 

} 

return N; // not found. 


public static void main(String[] args) // See page 769. 

} 


The constructor in this substring search algorithm builds a table giving the rightmost occurrence in 
the pattern of each possible character. The search method scans from right to left in the pattern, skip¬ 
ping to align any character causing a mismatch with its rightmost occurrence in the pattern. 
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■ If this computation would not in¬ 
crease i, we just increment i instead, 
to make sure that the pattern always 
slides at least one position to the 
right. The bottom example in the fig¬ 
ure at right illustrates this situation. 
algorithm 5.7 is a straightforward imple¬ 
mentation of this process. Note that the 
convention of using -1 in the right[] 
array entries corresponding to characters 
that do not appear in the pattern uni¬ 
fies the first two cases (increment i by 
j - right[txt.charAt(i+j)]). 

The full Boyer-Moore algorithm takes 
into account precomputed mismatches of 
the pattern with itself (in a manner simi¬ 
lar to the KMP algorithm) and provides a 
linear-time worst-case guarantee (whereas 
algorithm 5.7 can take time proportional 
to NM in the worst case—see exercise 
5 . 3 . 19 ). We omit this computation because 
the mismatched character heuristic con¬ 
trols the performance in typical practical 
applications. 


basic idea i 



i+j 



i 



1 






N 

L 

E 

N 

E 

E 

D 

L 

E 




t 






j 


could do better with 

increments by j - right['N'] 
to line up text with N in pattern 

i 

1 


KMP-like table 




N 

l' 

E. 




N 

E 

E D L E 

reset j to M-l t 






j 

heuristic is no help 

i 



i+j 



1 



1 






E 

L 

E 

N 

E 

E 

D 

L 

E 




t 






i 



lining up text with rightmost E 




would shift pattern left 








E 

L 

E. 

NEE 

D 

L 

E 


could do better with 

so increment i by 1 

i 

1 




KMP-like table 



E 

L 

E. 


N 

E 

E 

D 

L E 




reset j to 

M-l f 




Mismatched character heuristic (mismatch in pattern) 


Property O. On typical inputs, substring search with the Boyer-Moore mismatched 
character heuristic uses ~N/M character compares to search for a pattern of length 
M in a text of length N. 

Discussion: This result can be proved for various random string models, but such 
models tend to be unrealistic, so we shall skip the details. In many practical situa¬ 
tions it is true that all but a few of the alphabet characters appear nowhere in the 
pattern, so nearly all compares lead to M characters being skipped, which gives the 
stated result. 
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Rabin-Karp fingerprint search The method developed by M. O. Rabin and 
R. A. Karp is a completely different approach to substring search that is based on hash¬ 
ing. We compute a hash function for the pattern and then look for a match by using 
the same hash function for each possible M-character substring of the text. If we find 
a text substring with the same hash value as the pattern, we can check for a match. 
This process is equivalent to storing the pattern in a hash table, then doing a search 
for each substring of the text, but we do not need to reserve the memory for the hash 
table because it would have just one entry. A straightforward implementation based 
on this description would be much slower than a brute-force search (since comput¬ 
ing a hash function that involves every character is likely to be much more expensive 
than just comparing characters), but Rabin and Karp showed that it is easy to compute 
hash functions for M-character substrings in constant time (after some preprocessing), 
which leads to a linear-time substring search in practical situations. 


Basic plan. A string of length M corresponds to an M-digit base-R number. To use a hash 
table of size Q for keys of this type, we need a hash function to convert an M-digit base-R 
number to an i nt value between 0 and Q-l. Modular hashing (see section 3 . 4 ) pro¬ 
vides an answer: take the remainder when dividing the number by Q. In practice, we use 
a random prime Q, taking as large a value as possible while avoiding overflow (because 
we do not actually need to store a hash table). The method is simplest to understand for 
small Q and R = 10, shown in the example below. To find the pattern 2 6 5 B 5 in the 
text 3141592653589793, we choose a table size Q (997 in the example), compute 
the hash value 26535 % 997 = 613, and then look for a match by computing hash val¬ 
ues for each five-digit substring 
in the text. In the example, we 
get the hash values 508, 201, 715, 
971, 442, and 929 before finding 
the match 613. 


pat.charAt(j) 
j 0 12 3 4 

2 6 5 3 5 % 997 = 613 

txt.charAt(i) 


i 

0 

1 

2 

3 

4 

5 

6 

7 

8 

9 10 11 12 13 

14 15 


3 

1 

4 

1 

5 

9 

2 

6 

5 

3 5 8 9 7 

9 3 

0 

3 

1 

4 

1 

5 

% 

997 

= 

508 



1 


1 

4 

1 

5 

9 

% 

997 

= 

201 


2 



4 

1 

5 

9 

2 

% 

997 

= 715 


3 




1 

5 

9 

2 

6 

% 

997 = 971 


4 





5 

9 

2 

6 

5 

% 997 = 442 














5 






9 

2 

6 

5 

3 % 997 = 929 

/ 

6 -* — return i = i 

5 



2 

6 

5 

3 5 % 997 = 

613 





Basis for Rabin-Karp substring search 



Computing the hash func¬ 
tion. With five-digit values, we 
could just do all the necessary 
calculations with i nt values, but 
what do we do when M is 100 or 
1,000? A simple application of 
Horner’s method, precisely like 
the method that we examined in 
section 3.4 for strings and other 
types of keys with multiple values, 
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leads to the code shown at right, which computes the hash function for an M-digit base- 
R number represented as a char array in time proportional to M. (We pass M as an argu¬ 
ment so that we can use the method for both the pattern and the text, as you will see.) 
For each digit in the number, we multiply by R, add the digit, and take the remainder 
when divided by Q. For example, computing the 
hash function for our pattern using this process 
is shown at the bottom of the page. The same 
method can work for computing the hash func¬ 
tions in the text, but the cost for the substring 
search would be a multiplication, addition, and 
remainder calculation for each text character, 
for a total of NM operations in the worst case, 
no improvement over the brute-force method. 

Key idea. The Rabin-Karp method is based on efficiently computing the hash func¬ 
tion for position i +1 in the text, given its value for position i. It follows directly from a 
simple mathematical formulation. Using the notation l t for txt. charAt (i), the num¬ 
ber corresponding to the M-character substring of txt that starts at position i is 

x, = f.F-i + uF-zf. . . + Wi«° 

and we can assume that we know the value of h(x ; ) = x i mod Q . Shifting one position 
right in the text corresponds to replacing x ; by 

U+i = (*; “ t t R M -' )R + t i+M . 


private long hash(String key, int M) 

{ // Compute hash for key[0..M-l]. 

long h = 0; 

for (int j = 0; j < M; j++) 

h = (R * h + key. charAt (j)) % Q; 
return h; 

} 

Horner's method, applied to modular hashing 


We subtract off the leading digit, multiply by R, then add the trailing digit. Now, the 
crucial point is that we do not have to maintain the values of the numbers, just the 
values of their remainders when divided by Q. A fundamental property of the modu¬ 
lus operation is that if we take the remainder when divided by Q after each arithmetic 
operation, then we get the same answer as if we were to perform all of the arithmetic 
operations, then take the remainder 
when divided by Q. We took advan¬ 
tage of this property once before, 
when implementing modular hash¬ 
ing with Horner’s method (see page 
460). The result is that we can effec¬ 
tively move right one position in the 
text in constant time, whether M is 5 
or 100 or 1,000. 


pat.charAt(j) 
12 3 4 


6 5 3 5 

% 997 = 2 ^ / 

6 % 997 = (2*10 + 6) % 997 = 26 

5 % 997 = (26*10 + 5) % 997 = 265 

5 3 % 997 = (265*10 + 3) % 997 = 659 
5 3 5 % 997 = (659*10 + 5) % 997 = 613 


Computing the hash value for the pattern with Horner's method 
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2 3 4 5 6 7 


current value 
new value 


Implementation. This discussion leads directly 
to the substring search implementation in al¬ 
gorithm 5 . 8 . The constructor computes a hash 
value patHash for the pattern; it also computes 
the value of R M-1 mod Q in the variable RM. The 
hashSearchO method begins by computing the 
hash function for the first M characters of the text 
and comparing that value against the hash value 
for the pattern. If that is not a match, it proceeds 
through the text string, using the technique above 
to maintain the hash function for the M charac¬ 
ters starting at position i for each i in a variable 
txtHash and comparing each new hash value to 

patHash. (An extra Q is added during the txtHash calculation to make sure that every¬ 
thing stays positive so that the remainder operation works as it should.) 


1 5 9 


15 9 2 


current value 

subtract leading digit 
multiply by radix 

add new trailing digit 
new value 


Key computation in Rabin-Karp substring search 
(move right one position in the text) 


A trick: Monte Carlo correctness. After finding a hash value for an M-character sub¬ 
string of txt that matches the pattern hash value, you might expect to see code to com¬ 
pare those characters with the pattern to ensure that we have a true match, not just a 
hash collision. We do not do that test because using it requires backup in the text string. 
Instead, we make the hash table “size” Q as large as we wish, since we are not actually 
building a hash table, just testing for a collision with one key, our pattern. We will use 
a 1 ong value greater than 10 20 , making the probability that a random key hashes to the 


01234567 


9 10 11 12 13 14 15 


0 

1 

2 

3 

4 

5 

6 

7 

8 
9 

10 


14 15 

% 997 = 3 


9 2 6 5 3 


5 8 
/ 


9 7 9 3 


% 997 = (3*10 + 1) % 997 = 31 
4 % 997 = (31*10 + 4) % 997 = 314 

1 % 997 = (314*10 + 1) % 997 = 150 


% 997 = (150*10 + 5) % 997 = 508 


RM 


% 997 = ((508 + 3*(997 - 30))*10 + 9) % 997 = 201 


return i -M+l = 6 


% 997 = ((201 + 1*(997 - 30))*10 + 2) % 997 = 715 
6 % 997 = ((715 + 4*(997 - 30))*10 + 6) % 997 = 971 

6 5 % 997 = ((971 + 1*(997 - 30))*10 + 5) % 997 = 442 match 

6 5 3 % 997 = ((442 + 5*(997 - 30))*10 + 3) % 997 = 929 j 

6 5 3 5 % 997 = ((929 + 9*(997 - 30))*10 + 5) % 997 = 613 


Rabin-Karp substring search example 
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ALGORITHM 5.8 Rabin-Karp fingerprint substring search 


public class RabinKarp 
{ 

private String pat; 

// 

private 

long patHash; 

// 

private 

int M; 

// 

private 

long Q; 

// 

private 

int R = 256; 

// 

private 

long RM; 

// 


pattern (only needed for Las Vegas) 

pattern hash value 

pattern length 

a large prime 

alphabet size 

RA(M-l) % Q 


public RabinKarpCString pat) 

{ 

this.pat = pat; // 

this.M = pat.lengthO ; 

Q = longRandomPrimeO ; 

RM = 1; 

for (int i =1; i <= M-l; 

RM = (R * RM) % Q; 
patHash = hash(pat, M); 


save pattern (only needed for Las Vegas) 

// See Exercise 5.3.33. 

i++) // Compute RA(M-l) % Q for use 

// in removing leading digit. 


public boolean check(int i) // Monte Carlo (See text.) 

{ return true; } // For Las Vegas, check pat vs txt(i..i-M+l). 

private long hash(String key, int M) 

// See text (page 775). 

private int search(String txt) 

{ // Search for hash match in text. 

int N = txt.lengthO; 
long txtHash = hash(txt, M); 

if (patHash == txtHash) return 0; // Match at beginning, 

for (int i = M; i < N; i++) 

{ // Remove leading digit, add trailing digit, check for match. 

txtHash = (txtHash + Q - RM-'txt.charAt(i-M) % Q) % Q; 
txtHash = (txtHash*R + txt.charAt(i)) % Q; 
if (patHash == txtHash) 

if (check(i - M + 1)) return i - M + 1; // match 

} 

return N; //no match found 

} 

} 


This substring search algorithm is based on hashing. It computes a hash value for the pattern in the 
constructor, then searches through the text looking for a hash match. 
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same value as our pattern less than 10 -20 , an exceedingly small value. If that value is not 
small enough for you, you could run the algorithms again to get a probability of fail¬ 
ure of less than 10 40 . This algorithm is an early and famous example of a Monte Carlo 
algorithm that has a guaranteed completion time but fails to output a correct answer 
with a small probability. The alternative method of checking for a match could be slow 
(it might amount to the brute-force algorithm, with a very small probability) but is 
guaranteed correct. Such an algorithm is known as a Las Vegas algorithm. 

Property P. The Monte Carlo version of Rabin-Karp substring search is linear-time 
and extremely likely to be correct, and the Las Vegas version of Rabin-Karp sub¬ 
string search is correct and extremely likely to be linear-time. 

Discussion: The use of the very large value of Q, made possible by the fact that we 
need not maintain an actual hash table, makes it extremely unlikely that a collision 
will occur. Rabin and Karp showed that when Q is properly chosen, we get a hash 
collision for random strings with probability 1/Q, which implies that, for practi¬ 
cal values of the variables, there are no hash matches when there are no substring 
matches and only one hash match if there is a substring match. Theoretically, a text 
position could lead to a hash collision and not a substring match, but in practice it 
can be relied upon to find a match. 


If your belief in probability theory (or in the random string model and the code we 
use to generate random numbers) is more half-hearted than resolute, you can add to 
checkO the code to check that the text matches the pattern, which turns algorithm 
5.8 into the Las Vegas version of the algorithm (see exercise 5 . 3 . 12 ). If you also add a 
check to see whether that code is ever executed, you might develop more faith in prob¬ 
ability theory as time wears on. 

rabin-karp substring search is known as a fingerprint search because it uses a small 
amount of information to represent a (potentially very large) pattern. Then it looks 
for this fingerprint (the hash value) in the text. The algorithm is efficient because the 
fingerprints can be efficiently computed and compared. 
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Summary The table at the bottom of the page summarizes the algorithms that we 
have discussed for substring search. As is often the case when we have several algo¬ 
rithms for the same task, each of them has attractive features. Brute-force search is easy 
to implement and works well in typical cases (Java’s i ndexOfO method in Stri ng uses 
brute-force search); Knuth-Morris-Pratt is guaranteed linear-time with no backup in 
the input; Boyer-Moore is sublinear (by a factor of M) in typical situations; and Rabin- 
Karp is linear. Each also has drawbacks: brute-force might require time proportional 
to MN; Knuth-Morris-Pratt and Boyer-Moore use extra space; and Rabin-Karp has a 
relatively long inner loop (several arithmetic operations, as opposed to character com¬ 
pares in the other methods). These characteristics are summarized in the table below. 


algorithm 

version 

operation count 

backup 

correct? 

extra 


guarantee 

typical 

in input? 


space 

brute force 

— 

MN 

1.1 N 

yes 

yes 

i 


full DFA 
(Algorithm 5.6) 

2 N 

IAN 

no 

yes 

MR 

Knuth-Morris-Pratt 

mismatch 

3 N 

IAN 

no 

yes 

M 


transitions only 




full algorithm 

3 N 

N/M 

yes 

yes 

R 

Boyer-Moore 

mismatched char 







heuristic only 
(Algorithm 5.7 ) 

MN 

N/M 

yes 

yes 

R 


Monte Carlo 

7N 

7N 


yes f 


Rabin-Karpt 

(Algorithm 5.8) 

no 

1 





Las Vegas 

7 N f 

7N 

yes 

yes 

1 


t probabilisitic guarantee, with uniform and independent hash function 

Cost summary for substring search implementations 
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| Q&A 

Q. This substring search problem seems like a bit of a toy problem. Do I really need to 
understand these complicated algorithms? 

A. Well, the factor of M speedup available with Boyer-Moore can be quite impressive in 
practice. Also, the ability to stream input (no backup) leads to many practical applica¬ 
tions for KMP and Rabin-Karp. Beyond these direct practical applications, this topic 
provides an interesting introduction to the use of abstract machines and randomiza¬ 
tion in algorithm design. 

Q. Why not simplify things by converting each character to binary, treating all text as 
binary text? 

A. That idea is not quite effective because of false matches across character boundaries. 
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EXERCISES 


5 . 3.1 Develop a brute-force substring search implementation Brute, using the same 
API as ALGORITHM 5 . 6 . 

5 . 3.2 Give the dfa[] [] array for the Knuth-Morris-Pratt algorithm for the pattern 
AAAAAAAAA, and draw the DFA, in the style of the figures in the text. 

5 . 3.3 Give the dfa[] [] array for the Knuth-Morris-Pratt algorithm for the pattern 
ABRACADABRA, and draw the DFA, in the style of the figures in the text. 

5 . 3.4 Write an efficient method that takes a string txt and an integer M as arguments 
and returns the position of the first occurrence of M consecutive blanks in the string, 
txt. 1 ength if there is no such occurrence. Estimate the number of character compares 
used by your method, on typical text and in the worst case. 

5 . 3.5 Develop a brute-force substring search implementation BruteForceRL that pro¬ 
cesses the pattern from right to left (a simplified version of algorithm 5 . 7 ). 

5 . 3.6 Give the ri ght [] array computed by the constractor in algorithm 5.7 for the 
pattern ABRACADABRA. 

5 . 3.7 Add to our brute-force implementation of substring search a count O method to 
count occurrences and a searchAl 1 () method to print all occurrences. 

5 . 3.8 Add to KMP a count () method to count occurrences and a searchAl 1 () method 
to print all occurrences. 

5 . 3.9 Add to BoyerMoore acountO method to count occurrences and a searchAl 1 () 
method to print all occurrences. 

5 . 3.10 Add to Rabi nKarp a count () method to count occurrences and a searchAl 1 0 
method to print all occurrences. 

5 . 3 . 11 Construct a worst-case example for the Boyer-Moore implementation in algo¬ 
rithm 5.7 (which demonstrates that it is not linear-time). 

5 . 3.12 Add the code to checkO in Rabi nKarp (algorithm 5 . 8 ) that turns it into a 
Las Vegas algorithm (check that the pattern matches the text at the position given as 
argument). 

5 . 3.13 In the Boyer-Moore implementation in algorithm 5 . 7 , show that you can set 
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ri ght [c] to the penultimate occurrence of c when c is the last character in the pattern. 

5 . 3.14 Develop versions of the substring search implementations in this section that 
use char [] instead of Stri ng to represent the pattern and the text. 

5 . 3.15 Design a brute-force substring search algorithm that scans the pattern from 
right to left. 

5 . 3 . 16 Show the trace of the brute-force algorithm in the style of the figures in the text 
for the following pattern and text strings 

a. pattern: AAAAAAAB text: AAAAAAAAAAAAAAAAAAAAAAAAB 

b. pattern: ABABABAB text: ABABABABAABABABABAAAAAAAA 

5 . 3.17 Draw the KMP DFA for the following pattern strings. 

a. AAAAAAB 

b. AACAAAB 

c. ABABABAB 

d. ABAABAAABAAAB 

e. ABAABCABAABCB 

5 . 3 . 18 Suppose that the pattern and text are random strings over an alphabet of size 
R (which is at least 2). Show that the expected number of character compares for the 
brute-force method is {N — M + 1) (1 — i?~ M ) / (1 — R _I ) £ 2 (N — M + 1). 

5 . 3.19 Construct an example where the Boyer-Moore algorithm (with only the mis¬ 
matched character heuristic) performs poorly. 

5 . 3.20 How would you modify the Rabin-Karp algorithm to determine whether any of 
a subset of k patterns (say, all of the same length) is in the text? 

Solution: Compute the hashes of the k patterns and store the hashes in a StringSET 
(see exercise 5 . 2 . 6 ). 

5 . 3.21 How would you modify the Rabin-Karp algorithm to search for a given pattern 
with the additional proviso that the middle character is a “wildcard” (any text character 
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at all can match it). 

5 . 3.22 How would you modify the Rabin-Karp algorithm to search for an H-by-V pat¬ 
tern in an N-by-N text? 

5 . 3.23 Write a program that reads characters one at a time and reports at each instant 
if the current string is a palindrome. Hint: Use the Rab in-Karp hashing idea. 
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5.3.24 Find all occurrences. Add a method findAll () to each of the four substring 
search algorithms given in the text that returns an Iterable<Integer> that allows 
clients to iterate through all offsets of the pattern in the text. 

5.3.25 Streaming. Add a search () method to KMP that takes variable of type In as 
argument, and searches for the pattern in the specified input stream without using any 
extra instance variables. Then do the same for Rabi nKarp. 

5.3.26 Cyclic rotation check. Write a program that, given two strings, determines 
whether one is a cyclic rotation of the other, such as exampl e and ampl eex. 

5.3.27 Tandem repeat search. A tandem repeat of a base string b in a string s is a 
substring of s having at least two consecutive copies b (nonoverlapping). Develop and 
implement a linear-time algorithm that, given two strings b and s, returns the index of 
the beginning of the longest tandem repeat of b in s. For example, your program should 
return B when b is abcab and s is abcabcababcababcababcab. 

5.3.28 Buffering in brute-force search. Add a search () method to your solution to 
exercise 5 . 3.1 that takes an input stream (of type In) as argument and searches for the 
pattern in the given input stream. Note: You need to maintain a buffer that can keep at 
least the previous M characters in the input stream. Your challenge is to write efficient 
code to initialize, update, and clear the buffer for any input stream. 

5.3.29 Buffering in Boyer-Moore. Add a searchO method to algorithm 5.7 that 
takes an input stream (of type In) as argument and searches for the pattern in the given 
input stream. 

5.3.30 Two-dimensional search. Implement a version of the Rabin-Karp algorithm to 
search for patterns in two-dimensional text. Assume both pattern and text are rect¬ 
angles of characters. 

5.3.31 Random patterns. How many character compares are needed to do a substring 
search for a random pattern of length 100 in a given text? 

Answer: None. The method 

public boolean search(char[] txt) 

{ return false; } 
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is quite effective for this problem, since the chances of a random pattern of length 100 
appearing in any text are so low that you may consider it to be 0 . 

5.3.32 Unique substrings. Solve exercise 5 . 2.14 using the idea behind the Rabin- 
Karp method. 

5.3.33 Random primes. Implement TongRandomPri me () for Rabi nKarp (algorithm 
5 . 8 ). Hint: A random n-digit number is prime with probability proportional to 1 In. 

5.3.34 Straight-line code. The Java Virtual Machine (and your computer’s assembly 
language) support a goto instruction so that the search can be “wired in” to machine 
code, like the program at right (which is exactly equiva¬ 
lent to simulating the DFA for the pattern as in KM Pd fa, 
but likely to be much more efficient). To avoid check¬ 
ing whether the end of the text has been reached each 
time i is incremented, we assume that the pattern it¬ 
self is stored at the end of the text as a sentinel, as the 
last M characters of the text. The goto labels in this code 
correspond precisely to the dfa[] array. Write a static 
method that takes a pattern as input and produces as 
output a straight-line program like this that searches for 
the pattern. 

5.3.35 Boyer-Moore in binary strings. The mismatched character heuristic does not 
help much for binary strings, because there are only two possibilities for characters that 
cause the mismatch (and these are both likely to be in the pattern). Develop a substring 
search class for binary strings that groups bits together to make “characters” that can 
be used exactly as in algorithm 5 . 7 . Note: If you take b bits at a time, then you need 
a ri ght [] array with 2 h entries. The value of b should be chosen small enough so that 
this table is not too large, but large enough that most h-bit sections of the text are not 
likely to be in the pattern—there are M~b +1 different h-bit sections in the pattern (one 
starting at each bit position from 1 through M~b+ 1), so we want M~b +1 to be sig¬ 
nificantly less than 2 b . For example, if you take 2 b to be about lg (4 M), then the ri ght [] 
array will be more than three-quarters filled with -1 entries, but do not let b become less 
than Ml 2, since otherwise you could miss the pattern entirely, if it were split between 
two Z;-bit text sections. 



int i = -1; 



sm: 

i++; 



sO: 

if (txt[i]) ! = 

'A' 

goto sm 

si: 

if Ctxt[i]) ! = 

'A' 

goto sO 

s2: 

if Ctxt[i]5 ! = 

' B' 

goto sO 

sB: 

if Ctxt[i]) ! = 

'A' 

goto s2 

s4: 

if Ctxt[i]) ! = 

'A' 

goto sO 

s5: 

if Ctxtti]) ! = 

'A' 

goto sB 


return i-8; 



Straight-line substring search for A A B A A A 
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5.3.36 Random text. Write a program that takes integers M and N as arguments, gener¬ 
ates a random binary text string of length N, then counts the number of other occur¬ 
rences of the last M bits elsewhere in the string. Note: Different methods maybe appro¬ 
priate for different values of M. 

5.3.37 KMPfor random text. Write a client that takes integers M, N, and T as input and 
runs the following experiment T times: Generate a random pattern of length M and a 
random text of length N, counting the number of character compares used by KMP to 
search for the pattern in the text. Instrument KMP to provide the number of compares, 
and print the average count for the T trials. 

5.3.38 Boyer-Moore for random text. Answer the previous exercise for BoyerMoore. 

5.3.39 Timings. Write a program that times the four methods for the task of searchng 
for the substring 

it is a far far better thing that i do than i have ever done 

in the text of Tale of Two Cities (tale.txt). Discuss the extent to which your results 
validate the hypthotheses about performance that are stated in the text. 


www.it-ebooks.info 




This page intentionally left blank 


www.it-ebooks.info 


5.4 REGULAR EXPRESSIONS 



in many applications, we need to do substring searching with somewhat less than 
complete information about the pattern to be found. A user of a text editor may wish 
to specify only part of a pattern, or to specify a pattern that could match a few different 
words, or to specify that any one of a number of patterns would do. A biologist might 
search for a genomic sequence satisfying certain conditions. In this section we will con¬ 
sider how pattern matching of this type can be done efficiently. 

The algorithms in the previous section fundamentally depend on complete specifi¬ 
cation of the pattern, so we have to consider different methods. The basic mechanisms 
we will consider make possible a very powerful string-searching facility that can match 
complicated M-character patterns in JV-character text strings in time proportional to 
MN in the worst case, and much faster for typical applications. 

First, we need a way to describe the patterns: a rigorous way to specify the kinds of 
partial-substring-searching problems suggested above. This specification needs to in¬ 
volve more powerful primitive operations than the “check if the i th character of the text 
string matches the jth character of the pattern” operation used in the previous section. 
For this purpose, we use regular expressions, which describe patterns in combinations 
of three natural, basic, and powerful operations. 

Programmers have used regular expressions for decades. With the explosive growth 
of search opportunities on the web, their use is becoming even more widespread. We 
will discuss a number of specific applications at the beginning of the section, not only 
to give you a feeling for their utility and power, but also to enable you to become more 
familiar with their basic properties. 

As with the KMP algorithm in the previous section, we consider the three basic op¬ 
erations in terms of an abstract machine that can search for patterns in a text string. 
Then, as before, our pattern-matching algorithm will construct such a machine and 
then simulate its operation. Naturally, pattern-matching machines are typically more 
complicated than KMP DFAs, but not as complicated as you might expect. 

As you will see, the solution we develop to the pattern-matching problem is inti¬ 
mately related to fundamental processes in computer science. For example, the method 
we will use in our program to perform the string-searching task implied by a given 
pattern description is akin to the method used by the Java system to transform a given 
Java program into a machine-language program for your computer. We also encounter 
the concept of nondeterminism, which plays a critical role in the search for efficient 
algorithms (see chapter 6). 
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Describing patterns with regular expressions We focus on pattern descrip¬ 
tions made up of characters that serve as operands for three fundamental operations. 
In this context, we use the word language specifically to refer to a set of strings (pos¬ 
sibly infinite) and the word pattern to refer to a language specification. The rules that 
we consider are quite analogous to familiar rules for specifying arithmetic expressions. 

Concatenation. The first fundamental operation is the one used in the last section. 
When we write A B, we are specifying the language {A B } that has one two-character 
string, formed by concatenating A and B. 

Or. The second fundamental operation allows us to specify alternatives in the pattern. 
If we have an or between two alternatives, then both are in the language. We will use the 
vertical bar symbol | to denote this operation. For example, A | B specifies the language 
{ A, B } and A | E | I | 0 | U specifies the language {A,E,I,0,U}. Concatenation has 
higher precedence than or, so A B | BCD specifies the language { A B, B C D }. 

Closure. The third fundamental operation allows parts of the pattern to be repeated 
arbitrarily. The closure of a pattern is the language of strings formed by concatenating 
the pattern with itself any number of times (including zero). We denote closure by 
placing a * after the pattern to be repeated. Closure has higher precedence than con¬ 
catenation, so A B * specifies the language consisting of strings with an A followed by 0 
or more Bs, while A * B specifies the language consisting of strings with 0 or more As 
followed by a B. The empty string, which we denote by e, is found in every text string 
(and in A*). 

Parentheses. We use parentheses to override the default precedence rules. For exam¬ 
ple, C ( A C | B ) D specifies the language {CACD,CBD}; (A|C)((B|C)D) speci¬ 
fies the language { A B D, CBD, ACD, CCD }; and ( AB ) * specifies the language of 
strings formed by concatenating 
any number of occurrences of 
A B, including no occurrences: 

{ e, AB, ABAB, ...}. 

These simple rules allow us 
to write down REs that, while 
complicated, clearly and com¬ 
pletely describe languages (see 
the table at right for a few ex¬ 
amples). Often, a language can be simply described in some other way, but discovering 
such a description can be a challenge. For example, the RE in the bottom row of the 
table specifies the subset of ( A | B ) * with an even number of Bs. 


RE matches does not match 

( A | B ) ( C | D ) AC AD BC BD every other string 

A(B|C)*D AD ABD ACD ABCCBD BCD ADD ABCBC 

A* | (A*BA*BA*) * AAA BBAABB BABAAA ABA BBB BABBAAA 
Examples of regular expressions 
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regular expressions are extremely simple formal objects, even simpler than the 
arithmetic expressions that you learned in grade school. Indeed, we will take advantage 
of their simplicity to develop compact and efficient algorithms for processing them. 
Our starting point will be the following formal definition: 

Definition. A regular expression (RE) is either 

■ Empty 

■ A single character 

■ A regular expression enclosed in parentheses 

■ Two or more concatenated regular expressions 

■ Two or more regular expressions separated by the or operator (|) 

■ A regular expression followed by the closure operator (*) 


This definition describes the syntax of regular expressions, telling us what constitutes 
a legal regular expression. The semantics that tells us the meaning of a given regular 
expression is the point of the informal descriptions that we have given in this section. 
For review, we summarize these by continuing the formal definition: 

Definition (continued). Each RE represents a set of strings, defined as follows: 

■ The empty RE represents the empty set of strings, with 0 elements. 

■ A character represents the set of strings with one element, itself. 

■ An RE enclosed in parentheses represents the same set of strings as the RE 
without the parentheses. 

■ The RE consisting of two concatenated REs represents the cross product of 
the sets of strings represented by the individual components (all possible 
strings that can be formed by taking one string from each and concatenating 
them, in the same order as the REs). 

■ The RE consisting of the or of two REs represents the union of the sets rep¬ 
resented by the individual components. 

■ The RE consisting of the closure of an RE represents e (the empty string) 

or the union of the sets represented by the concatenation of any number of 
copies of the RE. 


In general, the language described by a given RE might be very large, possibly infinite. 
There are many different ways to describe each language: we must try to specify succinct 
patterns just as we try to write compact programs and implement efficient algorithms. 
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Shortcuts Typical applications adopt various additions to these basic rules to en¬ 
able us to develop succinct descriptions of languages of practical interest. From a theo¬ 
retical standpoint, these are each simply a shortcut for a sequence of operations involv¬ 
ing many operands; from a practical standpoint, they are a quite useful extention to the 
basic operations that enable us to develop compact patterns. 

Set-of-characters descriptors. It is often convenient 
to be able to use a single character or a short sequence 
to directly specify sets of characters. The dot character 
(.) is a wildcard that represents any single character. A 
sequence of characters within square brackets repre¬ 
sents any one of those characters. The sequence may 
also be specified as a range of characters. If preceded by 
a A, a sequence within square brackets represents any 
character but one of those characters. These notations 
are simply shortcuts for a sequence of or operations. 

Closure shortcuts. The closure operator specifies any 
number of copies of its operand. In practice, we want the flexibility to specify the num¬ 
ber of copies, or a range on the number. In particular, we use the plus sign (+) to specify 
at least one copy, the question mark (?) to specify zero or one copy, and a count or a 
range within braces ({}) to specify a given number of copies. Again, these notations are 
shortcuts for a sequence of the basic concatenation, or, and closure operations. 

Escape sequences. Some characters, such as \, ., |, *, (, and ), are metacha¬ 

racters that we use to form regular expressions. We use escape sequences that begin 
with a backslash character \ separating metacharacters from characters in the alphabet. 
An escape sequence may be a \ followed by a single metacharacter (which represents 
that character). For example, \\ represents \. Other escape sequences represent special 
characters and whitespace. For example, \t represents a tab character, \n represents a 
newline, and \s represents any whitespace character. 


option 

notation 

example 

shortcut for 

in language 

not in language 

at least 1 

+ 

(AB) + 

(AB) (AB)* 

AB ABABAB 

e BBBAAA 

0 or 1 

? 

(AB)? 

e | AB 

e AB 

any other string 

specific 

count in {} 

CAB){3} 

(AB)(AB)(AB) 

ABABAB 

any other string 

range 

range in {} 

(AB){1-2} 

(AB) | (AB) (AB) 

AB ABAB 

any other string 


Closure shortcuts (for specifying the number of copies of the operand) 


name 

notation 

example 

wildcard 


A. B 

specified set 

enclosed in [] 

[AEIOU] * 

range 

enclosed in [] 
separated by - 

[A-Z] 

[0-9] 

complement 

enclosed in [] 
preceded by A 

[AAEIOU]* 


Set-of-characters descriptors 


www.it-ebooks.info 




792 


CHAPTERS ■ Strings 


REs in applications REs have proven to be remarkably versatile in describing lan¬ 
guages that are relevant in practical applications. Accordingly, REs are heavily used and 
have been heavily studied. To familiarize you with regular expressions while at the same 
time giving you some appreciation for their utility, we consider a number of practical 
applications before addressing the RE pattern-matching algorithm. REs also play an 
important role in theoretical computer science. Discussing this role to the extent it 
deserves is beyond the scope of this book, but we sometimes allude to relevant funda¬ 
mental theoretical results. 

Substring search. Our general goal is to develop an algorithm that determines wheth¬ 
er a given text string is in the set of strings described by a given regular expression. If a 
text is in the language described by a pattern, we say that the text matches the pattern. 
Pattern matching with REs vastly generalizes the substring search problem of section 
5 . 3 . Precisely, to search for a substring pat in a text string txt is to check whether txt 
is in the language described by the pattern . * pat. * or not. 

Validity checking. You frequently encounter RE matching when you use the web. 
When you type in a date or an account number on a commercial website, the input¬ 
processing program has to check that your response is in the right format. One ap¬ 
proach to performing such a check is to write code that checks all the cases: if you were 
to type in a dollar amount, the code might check that the first symbol is a $, that the 
$ is followed by a set of digits, and so forth. A better approach is to define an RE that 
describes the set of all legal inputs. Then, checking whether your input is legal is pre¬ 
cisely the pattern-matching problem: is your input in the language described by the RE? 
Libraries of REs for common checks have sprung up on the web as this type of checking 
has come into widespread use. Typically, an RE is a much more precise and concise ex¬ 
pression of the set of all valid strings than would be a program that checks all the cases. 


context 

substring search 
phone number 
Java identifier 
genome marker 

email address 


regular expression matches 

.*NEEDLE.* A HAYSTACK NEEDLE 

\([0-9]{3}\)\ [0-9]{3}-[0-9]{4} (800) 867-5309 

[$_A-Za-z][$_A-Za-z0-9]* Pattern„Matcher 

gcg(cgg|agg)*ctg gcgaggaggcggcggctg 

[a-z]+@( [a-z]+\. ) + (edu | com) rs@cs. pH nceton . edu 

Typical regular expressions in applications (simplified versions) 
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Programmer’s toolbox. The origin of regular expression pattern matching is the Unix 
command grep, which prints all lines matching a given RE. This capability has proven 
invaluable for generations of programmers, and REs are built into many modern pro¬ 
gramming systems, from awk and emacs to Perl, Python, and JavaScript. For example, 
suppose that you have a directory with dozens of . j ava files, and you want to know 
which of them has code that uses Stdln. The command 

% grep Stdln *.java 

will immediately give the answer. It prints all lines that match . -'Stdln . * for each file. 

Genomics. Biologists use REs to help address important scientific problems. For 
example, the human gene sequence has a region that can be described with the RE 
gcg(cgg) *ctg, where the number of repeats of the egg pattern is highly variable 
among individuals, and a certain genetic disease that can cause mental retardation and 
other symptoms is known to be associated with a high number of repeats. 

Search. Web search engines support REs, though not always in their full glory. Typi¬ 
cally, if you want to specify alternatives with | or repetition with *, you can do so. 

Possibilities. A first introduction to theoretical computer science is to think about the 
set of languages that can be specified with an RE. For example, you might be surprised 
to know that you can implement the modulus operation with an RE: for example, 
(0 | 1 ( 01 * 0 )* 1 )* describes all strings of Os and Is that are the binary repre- 

sentatons of numbers that are multiples of three (!): 11,110,1001, and 110 0 are 
in the language, but 10,1011, and 10000 are not. 

Limitations. Not all languages can be specified with REs. A thought-provoking ex¬ 
ample is that no RE can describe the set of all strings that specify legal REs. Simpler 
versions of this example are that we cannot use REs to check whether parentheses are 
balanced or to check whether a string has an equal number of As and Bs. 

these examples just scratch the surface. Suffice it to say that REs are a useful part 
of our computational infrastructure and have played an important role in our under¬ 
standing of the nature of computation. As with KMP, the algorithm that we describe 
next is a byproduct of the search for that understanding. 
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Nondeterministic finite-state automata Recall that we can view the Knuth- 
Morris-Pratt algorithm as a finite-state machine constructed from the search pattern 
that scans the text. For regular expression pattern matching, we will generalize this idea. 

The finite-state automaton for KMP changes from state to state by looking at a char¬ 
acter from the text string and then changing to another state, depending on the char¬ 
acter. The automaton reports a match if and only if it reaches the accept state. The al¬ 
gorithm itself is a simulation of the automaton. The characteristic of the machine that 
makes it easy to simulate is that it is deterministic: each state transition is completely 
determined by the next character in the text. 

To handle regular expressions, we consider a more powerful abstract machine. Be¬ 
cause of the or operation, the automaton cannot determine whether or not the pattern 
could occur at a given point by examining just one character; indeed, because of clo¬ 
sure, it cannot even determine how many characters might need to be examined before 
a mismatch is discovered. To overcome these problems, we will endow the automaton 
with the power of nondeterminism: when faced with more than one way to try to match 
the pattern, the machine can “guess” the right one! This power might seem to you to 
be impossible to realize, but we will see that it is easy to write a program to build a 
nondeterministic finite-state automaton (NFA) and to efficiently simulate its operation. 
The overview of our RE pattern matching algorithm is the nearly the same as for KMP: 

■ Build the NFA corresponding to the given RE. 

■ Simulate the operation of that NFA on the given text. 

Kleene’s Theorem, a fundamental result of theoretical computer science, asserts that 
there is an NFA corresponding to any given RE (and vice versa). We will consider a 
constructive proof of this fact that will demonstrate how to transform any RE into an 
NFA; then we simulate the operation of the NFA to complete the job. 

Before we consider how to build pattern-matching NFAs, we will consider an exam¬ 
ple that illustrates their properties and the basic rules for operating them. Consider the 
figure below, which shows an NFA that determines whether a text string is in the lan¬ 
guage described by the RE ( ( A * B | A C ) D ) . As illustrated in this example, the NFAs 
that we define have the following characteristics: 

■ The NFA corresponding to an RE of length M has exactly one state per pattern 
character, starts at state 0, and has a (virtual) accept state M. 



NFA corresponding to the pattern ( ( A * B | A C ) D ) 
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■ States corresponding to a character from the alphabet have an outgoing edge 
that goes to the state corresponding to the next character in the pattern (black 
edges in the diagram). 

■ States corresponding to the metacharacters (, ), |, and * have at least one outgo¬ 
ing edge (red edges in the diagram), which may go to any other state. 

■ Some states have multiple outgoing edges, but no state has more than one out¬ 
going black edge. 

By convention, we enclose all patterns in parentheses, so the first state corresponds to a 
left parenthesis and the final state corresponds to a right parenthesis (and has a transi¬ 
tion to the accept state). 

As with the DFAs of the previous section, we start the NFA at state 0, reading the 
first character of a text. The NFA moves from state to state, sometimes reading text 
characters, one at a time, from left to right. However, there are some basic differences 
from DFAs: 

■ Characters appear in the nodes, not the edges, in the diagrams. 

■ Our NFA recognizes a text string only after explicitly reading all its characters, 
whereas our DFA recognizes a pattern in a text without necessarily reading all 
the text characters. 

These differences are not critical—we have picked the version of each machine that is 
best suited to the algorithms that we are studying. 

Our focus now is on checking whether the text matches the pattern—for that, we 
need the machine to reach its accept state and consume all the text. The rules for mov¬ 
ing from one state to another are also different than for DFAs—an NFA can do so in 
one of two ways: 

■ If the current state corresponds to a character in the alphabet and the current 
character in the text string matches the character, the automaton can scan past 
the character in the text string and take the (black) transition to the next state. 
We refer to such a transition as a match transition. 

■ The automaton can follow any red edge to another state without scanning any 
text character. We refer to such a transition as an e-transition, referring to the 
idea that it corresponds to “matching” the empty string e. 


A 


A 


A 


A 


B 


D 




7 


/ 


/ 


match transition: 
scan to next input character 
and change state 


-transition: 
change state 
with no match 


accept state reached 
and all text characters scanned : 
NFA recognizes text 


Finding a pattern with ( ( A * B | A C ) D ) NFA 


www.it-ebooks.info 


796 


CHAPTER 5 


■ Strings 


0 


0 


0 


For example, suppose that our NFA for 
C(A*B|AC)D)is started (at state 0 ) 
with the text A A A A B D as input. The figure 
at the bottom of the previous page shows a se¬ 
quence of state transitions ending in the accept 
state. This sequence demonstrates that the text 
is in the set of strings described by the RE—the 
text matches the pattern. With respect to the 
NFA, we say that the NFA recognizes that text. 

The examples shown at left illustrate that it 
is also possible to find transition sequences that 
cause the NFA to stall, even for input text such 
as A A A A B D that it should recognize. For 
example, if the NFA takes the transition to state 
4 before scanning all the As, it is left with nowhere to go, since the only way out of state 
4 is to match a B. These two examples demonstrate the nondeterministic nature of the 
automaton. After scanning an A and finding itself in state 3 , the NFA has two choices: 
it could go on to state 4 or it could go back to state 2 . The choices make the difference 
between getting to the accept state (as in the first example just discussed) or stalling (as 
in the second example just discussed). This NFA also has a choice to make at state 1 
(whether to take an e-transition to state 2 or to state 6). 

These examples illustrate the key difference between NFAs and DFAs: since an NFA 
may have multiple edges leaving a given state, the transition from such a state is not 
deterministic —it might take one transition at one point in time and a different transi¬ 
tion at a different point in time, without scanning past any text character. To make 
some sense of the operation of such an automaton, imagine that an NFA has the power 
to guess which transition (if any) will lead to the accept state for the given text string. 
In other words, we say that an NFA recognizes a text string if and only if there is some 
sequence of transitions that scans all the text characters and ends in the accept state when 
started at the beginning of the text in state 0 . Conversely, an NFA does not recognize a 
text string if and only if there is no sequence of match transitions and e-transitions that 
can scan all the text characters and lead to the accept state for that string. 

As with DFAs, we have been tracing the operation of the NFA on a text string simply 
by listing the sequence of state changes, ending in the final state. Any such sequence is a 
proof that the machine recognizes the text string (there maybe other proofs). But how 
do we find such a sequence for a given text string? And how do we prove that there is no 
such sequence for another given text string? The answers to these questions are easier 
than you might think: we systematically try all possibilities. 


AAA 


wrong guess if input is 

A A A A B D 


A 

- 1 + 6+7 


no way out 
of state 7 


no way out 
of state 4 


-1+2+3+2+3+2+3+2+3+4 no way out 

of state 4 


Stalling sequences for ( ( A * B | A C ) D ) NFA 
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Simulating an NFA The idea of an automaton that can guess the state transitions 
it needs to get to the accept state is like writing a program that can guess the right an¬ 
swer to a problem: it seems ridiculous. On reflection, you will see that the task is con¬ 
ceptually not at all difficult: we make sure that we check all possible sequences of state 
transitions, so if there is one that gets to the accept state, we will find it. 

Representation. To begin, we need an NFA representation. The choice is clear: the 
RE itself gives the state names (the integers between 0 and M, where M is the number of 
characters in the RE). We keep the RE itself in an array re[] of char values that de¬ 
fines the match transitions (if re [i ] is in the alphabet, then there is a match transition 
from i to i + 1 ). The natural representation for the e-transitions is a digraph —they are 
directed edges (red edges in our diagrams) connecting vertices between 0 and M (one 
for each state). Accordingly, we represent all the e-transitions as a digraph C. We will 
consider the task of building the digraph associated with a given RE after we consider 
the simulation process. For our example, the digraph consists of the nine edges 

0-1 1-2 1-6 2-3 3-2 3-4 5-8 8-9 10-11 

NFA simulation and reachability. To simulate an NFA, we keep track of the set of 
states that could possibly be encountered while the automaton is examining the cur¬ 
rent input character. The key computation is the familiar multiple-source reachability 
computation that we addressed in algorithm 4.4 (page 571). To initialize this set, we find 
the set of states reachable via e-transitions from state 0. For each such state, we check 
whether a match transition for the first input character is possible. This check gives 
us the set of possible states for the NFA just after matching the first input character. 
To this set, we add all states that could be reached via e-transitions from one of the 
states in the set. Given the set of possible states for the NFA just after matching the first 
character in the input, the solution to the multiple-source reachability problem in the 
-transition digraph gives the set of states that could lead to match transitions for the 
second character in the input. For example, the initial set of states for our example NFA 
is 0 1 2 3 4 6 ; if the first character is an A, the NFA could take a match transition to 
3 or 7; then it could take e-transitions from 3 to 2 or 3 to 4, so the set of possible states 
that could lead to a match transition for the second character is 2 3 4 7. Iterating this 
process until all text characters are exhausted leads to one of two outcomes: 

■ The set of possible states contains the accept state. 

■ The set of possible states does not contain the accept state. 

The first of these outcomes indicates that there is some sequence of transitions that 
takes the NFA to the accept state, so we report success. The second of these outcomes 
indicates that the NFA always stalls on that input, so we report failure. With our SET 
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2 3 4 : set of states reachable via e-transitions after matching A A 


0 1 2 _ 3 4 5 




10 11 : set of states reachable via e-transitions after matching A A B D 

0 1 2 3 4 5 ^""6 8 9 10 11 

-(b)—K jJ ©““KD - O 

/ 

accept ! 



Simulation of ( ( A * B | A C ) D ) NFA for input A A B D 
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data type and the Di rectedDFS class just described for computing multiple-source 
reachability in a digraph, the NFA simulation code given below is a straightforward 
translation of the English-language description just given. You can check your under¬ 
standing of the code by following the trace on the facing page, which illustrates the full 
simulation for our example. 

Proposition Q. Determining whether an /V-character text string is recognized by 
the NFA corresponding to an M-character RE takes time proportional to NM in 
the worst case. 

Proof: For each of the N text characters, we iterate through a set of states of size 
no more than M and run a DFS on the digraph of e-transitions. The construction 
that we will consider next establishes that the number of edges in that digraph is no 
more than 2M, so the worst-case time for each DFS is proportional to M. 


Take a moment to reflect on this remarkable result. This worst-case cost, the product 
of the text and pattern lengths, is the same as the worst-case cost of finding an exact 
substring match using the el¬ 
ementary algorithm that we 

i , • • public boolean recognizes(Strinq txt) 

started with at the beginning r 

& & { / Does the NFA recognize txt? 

of SECTION 5 . 3 . Bag<Integer> pc = new Bag<Integer>() ; 

DirectedDFS dfs = new DirectedDFS(C, 0); 
for (int v = 0; v < G.V(); v++) 
if (dfs.marked(v)) pc.add(v); 

for (int i = 0; i < txt.length(); i++) 

{ // Compute possible NFA states for txt[i+l]. 

Bag<Integer> match = new Bag<Integer>(); 
for (int v : pc) 
if (v < M) 

if (re[v] == txt.charAt(i) || re[v] == '.') 
match.add(v+l); 
pc = new Bag<Integer>(); 
dfs = new DirectedDFS(G, match); 
for (int v = 0; v < G.V(); v++) 
if (dfs.marked(v)) pc.add(v); 

} 

for (int v : pc) if (v == M) return true; 
return false; 


NFA simulation for pattern matching 
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Building an NFA corresponding to an RE From the similarity between regu¬ 
lar expressions and familiar arithmetic expressions, you may not be surprised to find 
that translating an RE to an NFA is somewhat similar to the process of evaluating an 
arithmetic expression using Dijkstra’s two-stack algorithm, which we considered in 
section 1 . 3 . The process is a bit different because 

■ REs do not have an explicit operator for concatenation 

■ REs have a unary operator, for closure (*) 

■ REs have only one binary operator, for or (|) 

Rather than dwell on the differences and similarities, we will consider an implementa¬ 
tion that is tailored for REs. For example, we need only one stack, not two. 

From the discussion of the representation at the beginning of the previous subsec¬ 
tion, we need only build the digraph G that consists of all the e-transitions. The RE itself 
and the formal definitions that we considered at the beginning of this section provide 
precisely the information that we need. Taking a cue from Dijkstra’s algorithm, we will 
use a stack to keep track of the positions of left parentheses and or operators. 

Concatenation. In terms of the NFA, the concatenation operation is the simplest to 
implement. Match transitions for states corresponding to characters in the alphabet 
explicitly implement concatenation. 

Parentheses. We push the RE index of each left parenthesis on the stack. Each time we 
encounter a right parenthesis, we eventually pop the corresponding left parentheses 
from the stack in the manner described below. As in Dijkstra’s algorithm, the stack en¬ 
ables us to handle nested parentheses in a natural manner. 

Closure. A closure (*) operator must occur either (i ) after a single character, when we 
add e-transitions to and from the character, or (z’i) after a right parenthesis, when we 
add e-transitions to and from the corresponding left parenthesis, the one at the top of 
the stack. 

Or expression. We process an RE of the form (A | B) where A and B are both REs by 
adding two e-transitions: one from the state corresponding to the left parenthesis to the 
state corresponding to the first character of B and one from the state corresponding to 
the | operator to the state corresponding to the right parenthesis. We push the RE index 
corresponding the | operator onto the stack (as well as the index corresponding to the 
left parenthesis, as described above) so that the information we need is at the top of the 
stack when needed, at the time we reach the right parenthesis. These e-transitions allow 
the NFA to choose one of the two alternatives. We do not add an e-transition from the 
state corresponding to the | operator to the state with the next higher index, as we have 
for all other states—the only way for the NFA to leave such a state is to take a transition 
to the state corresponding to the right parenthesis. 
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these simple rules suffice to build NFAs corresponding to arbitrarily complicated 
REs. algorithm 5.9 is an implementation whose constructor builds the e-transition 
digraph corresponding to a given RE, and a trace of the construction for our example 
appears on the following page. You can find other examples at the bottom of this page 
and in the exercises and are encouraged to enhance your understanding of the process 
by working your own examples. For brevity and for clarity, a few details (handling 
metacharacters, set-of-character descriptors, closure shortcuts, and multiway or op¬ 
erations) are left for exercises (see exercises 5 . 4.16 through 5 . 4 . 21 ). Otherwise, the 
construction requires remarkably little code and represents one of the most ingenious 
algorithms that we have seen. 


single-character closure 



C.addEdge(i , i+1) ; 
G.addEdge(i+l, i); 


closure expression 



G.addEdgeOp, i+ 1 ); 
G.addEdge(i+l, lp); 


or expression 



G.addEdgeOp, or+1) ; 
G.addEdge(or, i); 


NFA construction rules 



NFA corresponding to the pattern (.*AB((C|D*E)F)*G) 
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ALGORITHM 5.9 Regular expression pattern matching (grep). 


public class NFA 

{ 

private char[] re; // match transitions 

private Digraph G; // epsilon transitions 

private int M; // number of states 

public NFA(String regexp) 

{ // Create the NFA for the given regular expression. 

Stack<Integer> ops = new Stack<Integer>(); 
re = regexp. toCharArrayO ; 

M = re.length; 

G = new Digraph(M+l); 

for (int i =0; i < M; i++) 

{ 

int 1p = i; 

if (re[i] == '(' || re[i] == '|') 
ops.push(i); 
else if (re[i] == ')') 

{ 

int or = ops.pop(); 
if (re[or] == '|') 

{ 

Ip = ops.popO ; 

G.addEdge(lp, or+1); 

G.addEdge(or, i); 

} 

else Ip = or; 

} 

if (i < M-l && re[i+l] == '*') // lookahead 

{ 

G.addEdge(lp, i+1); 

G.addEdge(i+l, Ip); 

} 

if (re[i] == '(' || re[i] == || re[i] == ')') 

G.addEdge(i, i+1) ; 

} 

} 

public boolean recognizesCString txt) 

// Does the NFA recognize txt? (See page 799.) 

} 


This constructor builds an NFA corresponding to a given RE by creating a digraph of e-transitions. 
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stack for 

indices of o 

left parentheses (7) —► 

andors \ 

(ops) 



Building the NFA corresponding to ( ( A * B | A C ) D ) 


www.it-ebooks.info 



























804 


CHAPTERS ■ Strings 


Proposition R. Building the NFA corresponding to an M-c haracter RE takes time 
and space proportional to M in the worst case. 

Proof. For each of the M RE characters in the regular expression, we add at most 
three e-transitions and perhaps execute one or two stack operations. 


The classic GREP client for pattern matching, illustrated in the code at left, takes an RE 
as argument and prints the lines from standard input having some substring that is 

in the language described by the RE. 
This client was a feature in the early 
implementations of Unix and has 
been an indispensable tool for gen¬ 
erations of programmers. 


Classic Generalized Regular Expression Pattern-matching NFA client 


public class GREP 
{ 

public static void main(String[] args) 

{ 

String regexp = + args[0] + 

NFA nfa = new NFA(regexp); 
while (Stdln.hasNextLine()) 

{ 

String txt = Stdln .hasNextLine() ; 
if (nfa.recognizes(txt)) 

StdOut.println(txt); 

} 

} 

} 


% more tinyL.txt 

AC 

AD 

AAA 

ABD 

ADD 

BCD 

ABCCBD 
BABAAA 
BABBAAA 

% java GREP "(A*B|AC)D" < tinyL.txt 
ABD 

ABCCBD 

% java GREP Stdln < GREP.java 

while (Stdln.hasNextLine()) 

String txt = Stdln.hasNextLine(); 
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| Q&A 

Q. What is the difference between null and e? 

A. The former denotes an empty set; the latter denotes an empty string. You can have a 
set that contains one element, e, and is therefore not null. 
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EXERCISES 


5.4.1 Give regular expressions that describe all strings that contain 

■ Exactly four consecutive As 

■ No more than four consecutive As 

■ At least one occurrence of four consecutive As 

5.4.2 Give a brief English description of each of the following REs: 

a. . * 

b. A. *A | A 

c. . *ABBABBA. * 

d. * A. *A. *A. *A. * 

5.4.3 What is the maximum number of different strings that can be described by a 
regular expression with M or operators and no closure operators (parentheses and con¬ 
catenation are allowed)? 

5.4.4 Draw the NFA corresponding to the pattern ( ( ( A | B ) * | CD* | E F G ) * ) * . 

5.4.5 Draw the digraph of e-transitions for the NFA from exercise 5 . 4 . 4 . 

5.4.6 Give the sets of states reachable by your NFA from exercise 5 . 4.4 after each 
character match and susbsequent e-transitions for the input ABBACEFGEFGCAAB. 

5.4.7 Modify the GREP client on page 804 to be a client GREPmatch that encloses the pat¬ 
tern in parentheses but does not add .* before and after the pattern, so that it prints 
out only those lines that are strings in the language described by the given RE. Give the 
result of typing each of the following commands: 

a. % java GREPmatch "(A|B)(C|D)" < tinyL.txt 

b. % java GREPmatch "A(B|C)*D" < tinyL.txt 

c. % java GREPmatch "(A*B|AC)D" < tinyL.txt 

5.4.8 Write a regular expression for each of the following sets of binary strings: 

a. Contains at least three consecutive Is 

b. Contains the substring 110 

c. Contains the substring 1101100 

d. Does not contain the substring 110 
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5.4.9 Write a regular expression for binary strings with at least two Os but not con¬ 
secutive Os. 

5.4.10 Write a regular expression for each of the following sets of binary strings: 

a. Has at least 3 characters, and the third character is 0 

b. Number of Os is a multiple of 3 

c. Starts and ends with the same character 

d. Odd length 

e. Starts with 0 and has odd length, or starts with 1 and has even length 
f Length is at least 1 and at most 3 

5.4.11 For each of the following regular expressions, indicate how many bitstrings of 
length exactly 1,000 match: 

a. 0(0 | 1)*1 

b. 0*101* 

c. (1 | 01 )* 

5.4.12 Write a Java regular expression for each of the following: 

a. Phone numbers, such as (609) 555-1234 

b. Social Security numbers, such as 123-45-6789 

c. Dates, such as December 31, 1999 

d. IP addresses of the form a. b. c. d where each letter can represent one, two, 
or three digits, such as 196.26.155.241 

e. License plates that start with four digits and end with two uppercase letters 


www.it-ebooks.info 


808 


CHAPTER 5 ■ Strings 


CREATIVE PROBLEMS 


5.4.13 Challenging REs. Construct an RE that describes each of the following sets of 
strings over the binary alphabet: 

a. All strings except 11 or 111 

b. Strings with 1 in every odd-number bit position 

c. Strings with at least two Os and at most one 1 

d. Strings with no two consecutive Is 

5.4.14 Binary divisibility. Construct an RE that describes all binary strings that when 
interpreted as a binary number are 

a. Divisible by 2 

b. Divisible by 3 

c. Divisible by 123 

5.4.15 One-level REs. Construct a Java RE that describes the set of strings that are 
legal REs over the binary alphabet, but with no occurrence of parentheses within pa¬ 
rentheses. For example, (0. *1)* or (1. *0)* is in this language, but (1(0 or 1)1)* 
is not. 

5.4.16 Multiway or. Add multiway or to NFA. Your code should produce the machine 
drawn below for the pattern (.*AB((C|D|E)F)*G). 
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5.4.17 Wildcard. Add to NFA the capability to handle wildcards. 

5.4.18 One or more. Add to NFA the capability to handle the + closure operator. 

5.4.19 Specified set. Add to NFA the capability to handle specified-set descriptors. 

5.4.20 Range. Add to NFA the capability to handle range descriptors. 

5.4.21 Complement. Add to NFA the capability to handle complement descriptors. 

5.4.22 Proof. Develop a version of NFA that prints a proof that a given string is in the 
language recognized by the NFA (a sequence of state transitions that ends in the accept 
state). 
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The world is awash with data, and algorithms designed to represent data efficiently play 
an important role in the modern computational infrastructure. There are two primary 
reasons to compress data: to save storage when saving information and to save time 
when communicating information. Both of these reasons have remained important 
through many generations of technology and are familiar today to anyone needing a 
new storage device or waiting for a long download. 

You have certainly encountered compression when working with digital images, 
sound, movies, and all sorts of other data. The algorithms we will examine save space 
by exploiting the fact that most data files have a great deal of redundancy: For example, 
text files have certain character sequences that appear much more often than others; 
bitmap files that encode pictures have large homogeneous areas; and files for the digital 
representation of images, movies, sound, and other analog signals have large repeated 
patterns. 

We will look at an elementary algorithm and two advanced methods that are widely 
used. The compression achieved by these methods varies depending on characteristics 
of the input. Savings of 20 to 50 percent are typical for text, and savings of 50 to 90 per¬ 
cent might be achieved in some situations. As you will see, the effectiveness of any data 
compression method is quite dependent on characteristics of the input. Note: Usually, 
in this book, we are referring to time when we speak of performance; with data com¬ 
pression we normally are referring to the compression they can achieve, although we 
will also pay attention to the time required to do the job. 

On the one hand, data-compression techniques are less important than they once 
were because the cost of computer storage devices has dropped dramatically and far 
more storage is available to the typical user than in the past. On the other hand, data- 
compression techniques are more important than ever because, since so much storage 
is in use, the savings they make possible are greater. Indeed, data compression has come 
into widespread use with the emergence of the internet, because it is a low-cost way to 
reduce the time required to transmit large amounts of data. 

Data compression has a rich history (we will only be providing a brief introduction 
to the topic), and contemplating its role in the future is certainly worthwhile. Every stu¬ 
dent of algorithms can benefit from studying data compression because the algorithms 
are classic, elegant, interesting, and effective. 
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Rules of the game All of the types of data that we process with modern computer 
systems have something in common: they are ultimately represented in binary. We can 
consider each of them to be simply a sequence of bits (or bytes). For brevity, we use the 
term bitstream in this section to refer to a sequence of bits and bytestream when we are 
referring to the bits being considered as a sequence of fixed-size bytes. A bitstream or a 
bytestream might be stored as a file on your computer, or it might be a message being 
transmitted on the internet. 

Basic model. Accordingly, our basic model for data compression is quite simple, hav¬ 
ing two primary components, each a black box that reads and writes bitstreams: 

■ A compress box that transforms a bitstream B into a compressed version C (B ) 

■ An expand box that transforms C ( B ) back into B 

Using the notation | B | to denote the number of bits in a bitstream, we are interested in 
minimizing the quantity | C{B ) | / \B\, which is known as the compression ratio. 

bitstream B 

I 0110110101 


Compress 




Expand 


compressed version C(B) 


1101011111 ... 


original bitstream B 

0110110101 ... 


Basic model for data compression 


This model is known as lossless compression —we insist that no information be lost, in 
the specific sense that the result of compressing and expanding a bitstream must match 
the original, bit for bit. Lossless compression is required for many types of files, such as 
numerical data or executable code. For some types of files (such as images, videos, or 
music), it is reasonable to consider compression methods that are allowed to lose some 
information, so the decoder only produces an approximation of the original file. Lossy 
methods have to be evaluated in terms of a subjective quality standard in addition to 
the compression ratio.We do not address lossy compression in this book. 

Reading and writing binary data A full description of how information is en¬ 
coded on your computer is system-dependent and is beyond our scope, but with a few 
basic assumptions and two simple APIs, we can separate our implementations from 
these details. These APIs, Bi naryStdln and Bi naryStdOut, are modeled on the Stdln 
and StdOut APIs that you have been using, but their purpose is to read and write bits, 
where Stdln and StdOut are oriented toward character streams encoded in Unicode. 
An i nt value on StdOut is a sequence of characters (its decimal representation); an i nt 
value on Bi naryStdOut is a sequence of bits (its binary representation). 
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Binary input and output. Most systems nowadays, including Java, base their I/O on 
8 -bit bytestreams, so we might decide to read and write bytestreams to match I/O for¬ 
mats with the internal representations of primitive types, encoding an 8 -bit char with 
1 byte, a 16-bit short with 2 bytes, a 32-bit i nt with 4 bytes, and so forth. Since bit- 
streams are the primary abstraction for data compression, we go a bit further to allow 
clients to read and write individual bits, intermixed with data of primitive types. The 
goal is to minimize the necessity for type conversion in client programs and also to take 
care of operating system conventions for representing data. We use the following API 
for reading a bitstream from standard input: 

public class BinaryStdln 

boolean readBool ean() read 1 bit of data and return as a bool ean value 

char readCharQ read 8 bits of data and return as a char value 

u ^ ■, read r (between 1 and 16) bits of data 

char readChar(int r) v J 

and return as a char value 

[similar methods for byte (8 bits); short (16 bits); int (32 bits); long and double (64 bits)] 
boolean is Empty () is the bitstream empty? 

voi d cl ose O close the bitstream 

API for static methods that read from a bitstream on standard input 

A key feature of the abstraction is that, in marked constrast to Stdln, the data on stan¬ 
dard input is not necessarily aligned on byte boundaries. If the input stream is a sin¬ 
gle byte, a client could read it 1 bit at a time with eight calls to readBool ean (). The 
cl ose O method is not essential, but, for clean termination, clients should call cl ose () 
to indicate that no more bits are to be read. As with Stdln/StdOut, we use the follow¬ 
ing complementary API for writing bitstreams to standard output: 

public class BinaryStdOut 

void write (bool ean b) write the specified bit 
void write (char c) write the specified 8-bit char 

. , . , , . . write the r (between 1 and 16) least significant bits 

void write(char c, int r) ' ' 6 J 

of the specified char 

[similar methods for byte (8 bits); short (16 bits); int (32 bits); long and double (64 bits)] 
voi d cl ose () close the bitstream 

API for static methods that write to a bitstream on standard output 
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For output, the closeC) method is essential: clients must call close() to ensure that 
all of the bits specified in prior write () calls make it to the bitstream and that the final 
byte is padded with Os to byte-align the output for compatibility with the file system. 
As with the In and Out APIs associated with Stdln and StdOut, we also have available 
Bi naryln and Bi naryOut that allows us to reference binary-encoded files directly. 

Example. As a simple example, suppose that you have a data type where a date is rep¬ 
resented as three i nt values (month, day, year). Using StdOut to write those values in 
the format 12/31/1999 requires 10 characters, or 80 bits. If you write the values directly 
with Bi naryStdOut, you would produce 96 bits (32 bits for each of the 3 i nt values); 
if you use a more economical representation that uses byte values for the month and 
day and a short value for the year, you would produce 32 bits. With Bi naryStdOut you 
could also write a 4-bit field, a 5-bit field, and a 12-bit field, for a total of 21 bits (24 
bits, actually, because files must be an integral number of 8-bit bytes, so close() adds 
three 0 bits at the end). Important note: Such economy, in itself, is a crude form of data 
compression. 

Binary dumps. How can we examine the contents of a bitstream or a bytestream while 
debugging? This question faced early programmers when the only way to find a bug was 
to examine each of the bits in memory, and the term dump has been used since the early 
days of computing to describe a human-readable view of a bitstream. If you try to open 


a character stream (StdOut) 

StdOut.print(month + "/" + day + "/" + year); 

| ooiioooiooiiooioooioiiiiooiioiiiooiioooiooioiiiiooiioooijooiiiooijooiiiooiooiiiooi 

I 2 / 3 I 7 I /9 9 9 

/ 80 bits 

8-bit ASCII representation of' 9' 

32-bit integer representation of 31 

_z___ 

00000000|00000000|00000000|00001100|00000000|00000000000000000001111100000000000000000000011l|ll001111 

n ^ ^ 96 bits 


three ints (BinaryStdOut) 

BinaryStdOut.write(month); 
BinaryStdOut.write(day); 

BinaryStdOut.write(year); 


two chars and a short (Bi naryStdOut) 


a 4-bit field, a 5-bit field, and a 12-bit field (Bi naryStdOut) 


BinaryStdOut.wri te((char) month); 
BinaryStdOut.write((char) day); 

BinaryStdOut.write((short) year); 


BinaryStdOut.write(month, 4); 
BinaryStdOut.write(day, 5); 

BinaryStdOut.write(year, 12); 


00001100000111110000011111001111 
n 31 1999 


32 bits 


110011111011111001111000 

~"l2 31 1999 


21 bits ( + 3 bits for byte alignment at close) 


Four ways to put a date onto standard output 
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a file with an editor or view it in 
the manner in which you view text 
files (or just run a program that 
uses Bi naryStdOut), you are like¬ 
ly to see gibberish, depending on 
the system you use. Bi naryStdln 
allows us to avoid such system de¬ 
pendencies by writing our own 
programs to convert bitstreams 
such that we can see them with 
our standard tools. For example, 
the program Bi naryDump at left is 
a Bi naryStdln client that prints 
out the bits from standard in¬ 
put, encoded with the characters 
0 and 1. This program is useful 
for debugging when working with 
small inputs. The similar client 
HexDump groups the data into 8-bit bytes and prints each as two hexadecimal digits 
that each represent 4 bits. The client Pi ctureDump displays the bits in a Pi cture with 
0 bits represented as white pixels and 1 bits represented as black pixels. This pictorial 
representation is often useful in identifying patterns in a bitstream. You can download 
Bi naryDump, HexDump, and Pi ctureDump from the booksite. Typically, we use piping 
and redirection at the command-line level when working with binary files: we can pipe 
the output of an encoder to Bi naryDump, HexDump, or Pi ctureDump, or redirect it to a 
file. 

standard character stream 

% more abra.txt 
ABRACADABRA! 

bitstream represented as 0 and 1 characters 


bitstream represented as pixels in a Picture 


Four ways to look at a bitstream 


% java PictureDump 16 6 < abra.txt 

I ® L I _ 16-by-6 pixel 

I B window, magnified 

96 bits 


% java BinaryDump 16 < abra.txt 
0100000101000010 
0101001001000001 
0100001101000001 
0100010001000001 
0100001001010010 
0100000100100001 
96 bits 


bitstream represented with hex digits 

% java HexDump 4 < abra.txt 

41 42 52 41 
43 41 44 41 

42 52 41 21 
96 bits 


public class Bi naryDump 
{ 

public static void mai n(String[] args) 

{ 

int width = Integer.parselnt(args[0]); 
int cnt; 

for (cnt = 0; !BinaryStdln.isEmptyO; cnt++) 
{ 

if (width == 0) continue; 
if (cnt != 0 && cnt % width == 0) 

StdOut.println() ; 
if (BinaryStdln. readBooleanO) 

StdOut. pri nt("l") ; 
else StdOut. pri nt("0") ; 

} 

StdOut. println() ; 

StdOut.println(cnt + " bits"); 

} 

} 

Printing a bitstream on standard (character) output 
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012B456789ABCDEF 


ASCII encoding. When you HexDump a bit- 
stream that contains ASCII-encoded charac¬ 
ters, the table at right is useful for reference. 

Given a two digit hex number, use the first 
hex digit as a row index and the second hex 
digit as a column index to find the character 
that it encodes. For example, 31 encodes the 
digit 1, 4A encodes the letter 3, and so forth. 

This table is for 7-bit ASCII, so the first hex 
digit must be 7 or less. Hex numbers starting 
with 0 and 1 (and the numbers 20 and 7F) 
correspond to non-printing control charac¬ 
ters. Many of the control characters are left over from the days when physical devices 
such as typewriters were controlled by ASCII input; the table highlights a few that you 
might see in dumps. For example, SP is the space character, NUL is the null character, LF 
is line feed, and CR is carriage return. 


NUL 

SOH 

STX 

ETX 

EOT 

ENQ 

ACK 

BEL 

BS 

HT 

LF 

VT 

FF 

CR 

so 

SI 

DLE 

DC1 

DC2 

DC 3 

DC 4 

NAK 

SYN 

ETB 

CAN 

EM 

SUB 

ESC 

FS 

CS 

RS 

us 

SP 

i 

" 

# 

$ 

% 

& 

< 

( 

) 

* 

+ 

i 

- 


/ 

0 

i 

2 

3 

4 

5 

6 

7 

8 

9 


i 

< 

= 

> 

? 

@ 

A 

B 

C 

D 

E 

F 

C 

H 

I 

j 

K 

L 

M 

N 

0 

p 

Q 

R 

S 

T 

U 

V 

w 

X 

Y 

z 

[ 

\ 

] 

A 

_ 


a 

b 

C 

d 

e 

f 

g 

h 

i 

j 

k 

1 

m 

n 

0 

p 

q 

r 

s 

t 

u 

V 

W 

X 

y 

Z 

{ 

1 

> 

~ 

DEL. 


Hexadecimal-to-ASCII conversion table 


in summary, working with data compression requires us to reorient our thinking about 
standard input and standard output to include binary encoding of data. Bi naryStdln 
and Bi naryStdOut provide the methods that we need. They provide a way for you to 
make a clear distinction in your client programs between writing out information in¬ 
tended for file storage and data transmission (that will be read by programs) and print¬ 
ing information (that is likely to be read by humans). 
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Limitations To appreciate data-compression algorithms, you need to understand 
fundamental limitations. Researchers have developed a thorough and important theo¬ 
retical basis for this purpose, which we will consider briefly at the end of this section, 
but a few ideas will help us get started. 

Universal data compression. Armed with the algorithmic tools that have proven so 
useful for so many problems, you might think that our goal should be universal data 
compression: an algorithm that can make any bitstream smaller. Quite to the contrary, 
we have to adopt more modest goals because universal data compression is impossible. 


Proposition S. No algorithm can compress every bitstream. 

Proof: We consider two proofs that each provide some insight. 
The first is by contradiction: Suppose that you have an algorithm 
that does compress every bitstream. Then you could use that algo¬ 
rithm to compress its output to get a still shorter bitstream, and 
continue until you have a bistream of length 0! The conclusion 
that your algorithm compresses every bitstream to 0 bits is absurd, 
and so is the assumption that it can compress every bitstream. 

The second proof is a counting argument. Suppose that you have 
an algorithm that claims lossless compression for every 1,000-bit 
stream. That is, every such stream must map to a different shorter 
one. But there are only 1 + 2 + 4 + ... + 2 998 + 2 999 = 2 1000 —1 bit- 
streams with fewer than 1,000 bits and 2 1000 bitstreams with 1,000 
bits, so your algorithm cannot compress all of them. This argu¬ 
ment becomes more persuasive if we consider stronger claims. Say 
your goal is to achieve better than a 50 percent compression ratio. 
You have to know that you will be successful for only about 1 out 
of 2 500 of the 1,000-bit bitstreams! 



Put another way, you have at most a 1 in 2 500 chance of being able to 
compress by half a random 1,000-bit stream with any data-compres¬ 
sion algorithm. When you run across a new lossless compression al¬ 
gorithm, it is a sure bet that it will not achieve significant compression 
for a random bitstream. The insight that we cannot hope to compress 
random streams is a start to understanding data compression. We 
regularly process strings of millions or billions of bits but will never 
process even the tiniest fraction of all possible such strings, so we need 


I 


Universal 

data compression? 
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% java RandomBits | java PictureDump 2000 500 



1000000 bits 

A difficult file to compress: 1 million (pseudo-) random bits 


not be discouraged by this theoretical result. Indeed, the bitstrings that we regularly 
process are typically highly structured, a fact that we can exploit for compression. 


Undecidability. Consider the million-bit string pictured at the top of this page. This 
string appears to be random, so you are not likely to find a lossless compression algo¬ 
rithm that will compress it. But there is a way to represent that string with just a few 
thousand bits, because it was produced by the program below. (This program is an ex¬ 
ample of a pseudo-random number generator, like Java’s Math. random() method.) A 
compression algorithm that compresses by writing the program in ASCII and expands 
by reading the program and then running it achieves a .3 percent compression ratio, 
which is difficult to beat (and we can drive the ratio arbitrarily low by writing more 
bits). To compress such a file is to discover the program that produced it. This example 
is not so far-fetched as it first appears: when you compress a video or an old book 
that was digitized with a scanner or any of countless other types of files from the web, 
you are discovering something about the program that produced the file. The realiza¬ 
tion that much of the data that we process is 
produced by a program leads to deep issues 
in the theory of computation and also gives 
insight into the challenges of data compres¬ 
sion. For example, it is possible to prove that 
optimal data compression (find the short¬ 
est program to produce a given string) is an 
undecidable problem: not only can we not 
have an algorithm that compresses every bit- 
stream, but also we cannot have a strategy 
for developing the best algorithm! 


public class RandomBits 
{ 

public static void mai n(String [] args) 
{ 

int x = 11111; 

for (int i =0; i < 1000000; i++) 

{ 

x = x * 314159 + 218281; 

BinaryStdOut.write(x > 0); 

} 

BinaryStdOut.closeO ; 

} 

} 


A "compressed" million-bit stream 
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The practical impact of these limitations is that lossless compression methods must 
be oriented toward taking advantage of known structure in the bitstreams to be com¬ 
pressed. The four methods that we consider exploit, in turn, the following structural 
characteristics: 

■ Small alphabets 

■ Long sequences of identical bits/characters 

■ Frequently used characters 

■ Long reused bit/character sequences 

If you know that a given bitstream exhibits one or more of these characteristics, you 
can compress it with one of the methods that you are about to learn; if not, trying them 
each is probably still worth the effort, since the underlying structure of your data may 
not be obvious, and these methods are widely applicable. As you will see, each method 
has parameters and variations that may need to be tuned for best compression of a par¬ 
ticular bitstream. The first and last recourse is to learn something about the structure 
of your data yourself and exploit that knowledge to compress it, perhaps using one of 
the techniques we are about to consider. 
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Warmup: genomics As preparation for more complicated data-compression al¬ 
gorithms, we now consider an elementary (but very important) data-compression task. 
All of our implementations will use the same conventions that we will now introduce 
in the context of this example. 

Genomic data. As a first example of data compression, consider this string: 

ATAGATGCATACCCCATAGCTACATGTCCTAGCAT 

Using standard ASCII encoding (1 byte, or 8 bits per character), this string is a bitstream 
of length 8X35 = 280. Strings of this sort are extremely important in modern biology, 
because biologists use the letters A, C, T, and G to represent the four nucleotides in the 
DNA of living organisms. A genome is a sequence of nucleotides. Scientists know that 
understanding the properties of genomes is a key to understanding the processes that 
manifest themselves in living organisms, in¬ 
cluding life, death, and disease. Genomes for 
many living things are known, and scientists 
are writing programs to study the structure of 
these sequences. 

2-bit code compression. One simple prop¬ 
erty of genomes is that they contain only 
four different characters, so each can be en¬ 
coded with just 2 bits per character, as in the 
compress () method shown at right. Even 
though we know the input stream to be 
character-encoded, we use BinaryStdln to 
read the input, to emphasize adherence to the 
standard data-compression model (bitstream 
to bitstream). We include the number of encoded characters in the compressed file, to 
ensure proper decoding if the last bit does not fall at the end of a byte. Since it converts 
each 8-bit character to a 2-bit code and just adds 32 bits for the length, this program 
approaches a 25 percent compression ratio as the number of characters increases. 

2-bit code expansion. The expand () method at the top of the next page expands a 
bitstream produced by this compress () method. As with compression, this method 
reads a bitstream and writes a bitstream, in accordance with the basic data-compression 
model. The bitstream that we produce as output is the original input. 


public static void compress() 

{ 

Alphabet DNA = new A1phabet("ACTG"); 
String s = Bi naryStdln . readStri ng() ; 
int N = s . 1ength(); 
BinaryStdOut.write(N) ; 
for (int i =0; i < N; i++) 

{ // Write two-bit code for char, 

int d = DNA.toIndex(s.charAt(i)); 
BinaryStdOut.write(d, DNA.lgRO) ; 

} 

BinaryStdOut.closeO ; 


Compression method for genomic data 
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the same approach works for other fixed- 
size alphabets, but we leave this generaliza¬ 
tion for an (easy) exercise (see exercise 
5 . 5 . 25 ). 

These methods do not quite adhere to the 
standard data-compression model, because 
the compressed bitstream does not contain 
all the information needed to decode it. The 
fact that the alphabet is one of the letters A, 
C, T, or C is agreed upon by the two meth¬ 
ods. Such a convention is reasonable in an 
application such as genomics, where the 
same code is widely reused. Other situations 
might require including the alphabet in the encoded message (see exercise 5 . 5 . 25 ). 
The norm in data compression is to include such costs when comparing methods. 

In the early days of genomics, learning a genomic sequence was a long and arduous 
task, so sequences were relatively short and scientists used standard ASCII encoding to 
store and exchange them. The experimental process has been vastly streamlined, to the 
point where known genomes are numerous and lengthy (the human genome is over 
10 10 bits), and the 75 percent savings achieved by these simple methods is very signifi¬ 
cant. Is there room for further compression? That is a very interesting question to con¬ 
template, because it is a scientific question: the ability to compress implies the existence 
of some structure in the data, and a prime focus of modern genomics is to discover 
structure in genomic data. Standard data-compression methods like the ones we will 
consider are ineffective with ( 2 -bit-encoded) genomic data, as with random data. 

We package compress O and 
expand () as static methods in 
the same class, along with a sim¬ 
ple driver, as shown at right. To 
test your understanding of the 
rules of the game and the basic 
tools that we use for data com¬ 
pression, make sure that you 
understand the various com¬ 
mands on the facing page that 
invoke Genome.compress O and 
Genome. expand () on our sample 

data (and their consequences). Packaging convention for data-compression methods 


public class Genome 
{ 

public static void compressQ 

// See text. 

public static void expand() 

// See text. 

public static void main(String[] args) 

{ 

if (args[0] .equalsC"-")) compressQ; 
if (args[0] .equals("+")) expandO; 

} 


public static void expandO 
{ 

Alphabet DNA = new A1phabet("ACTG"); 
int w = DNA.lgRO ; 
int N = Bi naryStdln. readlntO ; 
for (int i = 0; i < N; i++) 

{ // Read 2 bits; write char. 

char c = BinaryStdln.readChar(w); 
BinaryStdOut.write(DNA.toChar(c)) ; 

} 

BinaryStdOut.closeO ; 

} 

Expansion method for genomic data 
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tiny test case (264 bits) 

% more genomeTiny.txt 
AT AGATGCATAGCGCAT ACCT AGAT GTGCTACC 


java BinaryDump 64 < genomeTiny.txt 
0100000101010100010000010100011101000001010101000100011101000011 
0100000101010100010000010100011101000011010001110100001101000001 
0101010001000001010001110100001101010100010000010100011101000001 
0101010001000111010101000100011101000011010101000100000101000111 
01000011 
264 bits 


% java Genome - < genomeTiny.txt 

?? -* - cannot see bitstream on standard output 

% java Genome - < genomeTiny.txt | java BinaryDump 64 
0000000000000000000000000010000100100011001011010010001101110100 
1000110110001100101110110110001101000000 
104 bits 

% java Genome - < genomeTiny.txt | java HexDump 8 
00 00 00 21 23 2d 23 74 
8d 8c bb 63 40 
104 bits 


% java Genome - < genomeTiny.txt > genomeTiny.2bit 
% java Genome + < genomeTiny.2bit 
AT AGATGCAT AGCGCAT AGCT AGATCT GCTAGC 

% java Genome - < genomeTiny.txt | java Gen 
AT AGAT GCAT AGCGCAT AGCT AGATGTGCTAGC -—- 


an actual virus (50000 bits) 

% java PictureDump 512 100 < genomeVirus.txt 



% java Genome - < genomeVirus.txt | java PictureDump 512 25 



12536 bits 


Compressing and expanding genomic sequences with 2-bit encoding 


www.it-ebooks.info 




















































































822 


CHAPTERS ■ Strings 


Run-length encoding The simplest type of redundancy in a bitstream is long 
runs of repeated bits. Next, we consider a classic method known as run-length encoding 
for taking advantage of this redundancy to compress data. For example, consider the 
following 40-bit string: 

0000000000000001111111000000011111111111 

This string consists of 15 0s, then 7 Is, then 7 0s, then 11 Is, so we can encode the bit¬ 
string with the numbers 15, 7, 7, and 11. All bitstrings are composed of alternating runs 
of 0s and Is; we just encode the length of the runs. In our example, if we use 4 bits to 
encode the numbers and start with a run of 0s, we get the 16-bit string 

1111011101111011 

(15 = 1111, then 7 = 0111, then 7 = 0111, then 11 = 1011) for a compression ratio of 
16/40 = 40 percent. In order to turn this description into an effective data compression 
method, we have to consider the following issues: 

■ How many bits do we use to store the counts? 

■ What do we do when encountering a run that is longer than the maximum 
count implied by this choice? 

■ What do we do about runs that are shorter than the number of bits needed to 
store their length? 

We are primarily interested in long bitstreams with relatively few short runs, so we ad¬ 
dress these questions by making the following choices: 

■ Counts are between 0 and 255, all encoded with 8 bits. 

■ We make all run lengths less than 256 by including runs of length 0 if needed. 

■ We encode short runs, even though doing so might lengthen the output. 

These choices are very easy to implement and also very effective for several kinds of bit- 
streams that are commonly encountered in practice. They are not effective when short 
runs are numerous—we save bits on a run only when the length of the run is more than 
the number of bits needed to represent itself in binary. 

Bitmaps. As an example of the effectiveness of run-length encoding, we consider bit¬ 
maps, which are widely use to represent pictures and scanned documents. For brev¬ 
ity and simplicity, we consider binary-valued bitmaps organized as bitstreams formed 
by taking the pixels in row-major order. To view the contents of a bitmap, we use 
Pi ctureDump. Writing a program to convert an image from one of the many common 
lossless image formats that have been defined for “screen shots” or scanned documents 
into a bitmap is a simple matter (see exercise 5 . 5 .x). Our example to demonstrate 
the effectiveness of run-length encoding comes from screen shots of this book: a let¬ 
ter q (at various resolutions). We focus on a binary dump of a 32-by-48-pixel screen 
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shot, shown at right along with run lengths for 
each row. Since each row starts and ends with 
a 0, there is an odd number of run lengths on 
each row; since the end of one row is followed 
by the beginning of the next, the corresponding 
run length in the bitstream is the sum of the last 
run length in each row and the first run length 
in the next (with extra additions corresponding 
to rows that are all 0). 

Implementation. The informal description 
just given leads immediately to the comp ress O 
and expand () implementations on the next 
page. As usual, the expand () implementation is 
the simpler of the two: read a run length, print 
that many copies of the current bit, complement 
the current bit, and continue until the input is 
exhausted. The compress () method is not 
much more difficult, consisting of the following 
steps while there are bits in the input stream: 

■ Read a bit. 

■ If it differs from the last bit read, write the 
current count and reset the count to 0. 

■ If it is the same as the last bit read, and 
the count is a maximum, write the count, 
write a 0 count, and reset the count to 0. 

■ Increment the count. 


% java BinaryDump 32 < q32x48.bin 

00000000000000000000000000 ojxr^oo 
ooooooooooooooooooooootmotfoooooo 
00000000 00000001111111(f000000000 
00000000000011111111111111100000 
00000000001111000011111111100000 
00000000111100000000011111100000 
00000001110000000000001111100000 
00000011110000000000001111100000 
00000111100000000000001111100000 
00001111000000000000001111100000 
00001111000000000000001111100000 
00011110000000000000001111100000 
00011110000000000000001111100000 
00111110000000000000001111100000 
00111110000000000000001111100000 
00111110000000000000001111100000 
00111110000000000000001111100000 
00111110000000000000001111100000 
00111110000000000000001111100000 
00111110000000000000001111100000 
00111110000000000000001111100000 
00111111000000000000001111100000 
00111111000000000000001111100000 
00011111100000000000001111100000 
00011111100000000000001111100000 
00001111110000000000001111100000 
00001111111000000000001111100000 
00000111111100000000001111100000 
00000011111111000000011111100000 
00000001111111111111111111100000 
00000000011111111111001111100000 
00000000000011111000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000001111100000 
00000000000000000000011111110000 
00000000000000000011111111111100 
00000000000000000111111111111110 
00 000000 OOOOOOOOOOIKLOOOOOO 000000 
00000000000000000000 Oft-QOOOOOOOOO 


1536 bits 


17 0s 


7 1s 


\ 


12 15 
10 4 


7 20 
9 11 


22 

22 

22 

22 

22 

22 

22 

22 

22 

22 

22 

22 

21 


18 12 
17 14 
32 
32 


A typical bitmap, with run lengths for each 


When the input stream empties, writing the 
count (length of the last run) completes the process. 


Increasing resolution in bitmaps. The primary reason that run-length encoding is 
widely used for bitmaps is that its effectiveness increases dramatically as resolution in¬ 
creases. It is easy to see why this is true. Suppose that we double the resolution for our 
example. Then the following facts are evident: 

■ The number of bits increases by a factor of 4. 

■ The number of runs increases by about a factor of 2. 

■ The run lengths increase by about a factor of 2. 

■ The number of bits in the compressed version increases by about a factor of 2. 

■ Therefore, the compression ratio is halved! 


10 

5 

4 9 5 

9 6 5 

12 5 5 

12 5 5 

13 5 5 

14 5 5 

14 5 5 

15 5 5 

15 5 5 

15 5 5 

15 5 5 

15 5 5 

15 5 5 

15 5 5 

15 5 5 

15 5 5 

15 5 5 

14 5 5 

14 5 5 

13 5 5 

13 5 5 
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public static void expand() 

{ 

boolean b = false; 

while (! BinaryStdln . isEmptyO) 

{ 

char cnt = BinaryStdln.readChar(); 
for (int i =0; i < cnt; i++) 
BinaryStdOut.write(b) ; 
b = ! b; 

} 

BinaryStdOut.closeO ; 

} 

public static void compress0 
{ 

char cnt = 0; 

boolean b, old = false; 

while (! Bi naryStdln . i sEmptyO) 

{ 

b = BinaryStdln.readBoolean(); 
if (b != old) 

{ 

BinaryStdOut.write(cnt) ; 
cnt = 0; 
old = !old; 

} 

el se 
{ 

if (cnt == 255) 

{ 

BinaryStdOut.write(cnt) ; 
cnt = 0; 

BinaryStdOut.write(cnt) ; 

} 

} 

cnt++; 

} 

BinaryStdOut.write(cnt); 
BinaryStdOut.closeO; 


Expand and compress methods for run-length encoding 


Without run-length encoding, space require¬ 
ments increase by a factor of 4 when the res¬ 
olution is doubled; with run-length encod¬ 
ing, space requirements for the compressed 
bitstream just double when the resolution is 
doubled. That is, space grows and the com¬ 
pression ratio drops linearly with resolution. 
For example, our (low-resolution) letter q 
yields just a 74 percent compression ratio; if 
we increase the resolution to 64 by 96, the ra¬ 
tio drops to 37 percent. This change is graph¬ 
ically evident in the PictureDump outputs 
shown in the figure on the facing page. The 
higher-resolution letter takes four times the 
space of the lower resolution letter (double 
in both dimensions), but the compressed ver¬ 
sion takes just twice the space (double in one 
dimension). If we further increase the resolu¬ 
tion to 128-by-192 (closer to what is needed 
for print), the ratio drops to 18 percent (see 
exercise 5 . 5 . 5 ). 

RUN-LENGTH ENCODING IS VERY EFFECTIVE 
in many situations, but there are plenty of 
cases where the bitstream we wish to com¬ 
press (for example, typical English-language 
text) may have no long runs at all. Next, we 
consider two methods that are effective for a 
broad variety of files. They are widely used, 
and you likely have used one or both of these 
methods when downloading from the web. 
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tiny test case (40 bits) 

% java BinaryDump 40 < 4runs.bin 
0000000000000001111111000000011111111111 
40 bits 

% java RunLength - < 4runs.bin | java HexDump 
Of 07 07 Ob 

32 bits compression ratio 32/40 — 80% 

% java RunLength - < 4runs.bin | java RunLength + | java BinaryDump 40 

0000000000000001111111000000011111111111 -*- compress-expand produces original input 

40 bits 


ASCII text (96 bits) 

% java RunLength - < abra.txt | java HexDump 24 

01 01 05 01 01 01 04 01 02 01 01 01 02 01 02 01 05 01 01 01 04 02 01 01 

05 01 01 01 03 01 03 01 05 01 01 01 04 01 02 01 01 01 02 01 02 01 05 01 

02 01 04 01 

416 bits -* - compression ratio 416/ — 433% — do not use run-length encoding for ASCII ! 


a bitmap (1536 bits) 

% java RunLength - < q32x48.bin > q32x48.bin.rie 
% java HexDump 16 < q32x48.bin.rle 
4f 07 16 Of Of 04 04 09 Od 04 09 06 Oc 03 Oc 05 

Ob 04 Oc 05 Oa 04 Od 05 09 04 Oe 05 09 04 Oe 05 

08 04 Of 05 08 04 Of 05 07 05 Of 05 07 05 Of 05 

07 05 Of 05 07 05 Of 05 07 05 Of 05 07 05 Of 05 

07 05 Of 05 07 05 Of 05 07 06 Oe 05 07 06 Oe 05 

08 06 Od 05 08 06 Od 05 09 06 Oc 05 09 07 Ob 05 

Oa 07 Oa 05 Ob 08 07 06 Oc 14 Oe Ob 02 05 11 05 

05 05 lb 05 lb 05 lb 05 lb 05 lb 05 lb 05 lb 05 

lb 05 lb 05 lb 05 lb 05 la 07 16 Oc 13 Oe 41 

1144 bi ts -*- compression ratio 1144/1536 = 74% 


a higher-resolution bitmap (6144 bits) 

% java BinaryDump 0 < q64x96.bin 
6144 bits 

% java RunLength - < q64x96.bin | java BinaryDump 0 
2296 bits -« - compression ratio 2296/6144 — 37% 


% java PictureDump 32 48 < q32x48.bin 



1536 bits 

% java PictureDump 32 36 < q32x48.rle.bin 



1144 bits 


% java PictureDump 64 96 < q64x96.bin 



6144 bits 


% java PictureDump 64 36 < q64x96.rle.bin 



Compressing and expanding bitstreams with run-length encoding 
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Huffman compression We now examine a data-compression technique that can 
save a substantial amount of space in natural language files (and many other kinds of 
files). The idea is to abandon the way in which text files are usually stored: instead of 
using the usual 7 or 8 bits for each character, we use fewer bits for characters that appear 
often than for those that appear rarely. 

To introduce the basic ideas, we start with a small example. Suppose we wish to en¬ 
code the string ABRACADABRA! Encoding it in 7-bit ASCII gives this bitstring: 

100000110000101010010100000110000111000001 - 

100010010000011000010101001010000010100001 . 

To decode this bitstring, we simply read off 7 bits at a time and convert according to 
the ASCII coding table on page 815. In this standard code the D, which appears only once, 
requires the same number of bits as the A, which appears five times. Huffman compres¬ 
sion is based on the idea that we can save bits by encoding frequently used characters 
with fewer bits than rarely used characters, thereby lowering the total number of bits 
used. 

Variable-length prefix-free codes. A code associates each character with a bitstring: a 
symbol table with characters as keys and bitstrings as values. As a start, we might try 
to assign the shortest bitstrings to the most commonly used letters, encoding A with 0, 
B with 1, R with 00, C with 01, D with 10, and ! with 11, so ABRACADABRA ! would 
be encoded as 0 1 00 0 01 0 10 0 1 00 0 11. This representation uses only 17 bits 
compared to the 77 for 7-bit ASCII, but it is not really a code because it depends on the 
blanks to delimit the characters. Without the blanks, the bitstring would be 

01000010100100011 

and could be decoded asCRRDDCRCB or as several other strings. Still, the count of 
17 bits plus 10 delimiters is rather more compact than the standard code, primarily be¬ 
cause no bits are used to encode letters not appearing in the message. The next step is to 
take advantage of the fact that delimiters are not needed if no character code is the prefix of 
another. A code with this property is known as a prefix-free code. The code just given is 
not prefix-free because 0, the code for A, is a prefix of 00, the code for R. For example, if 
we encode A with 0, B with 11111, C with 110, D with 100, R with 1110, and ! with 101, 
there is only one way to decode the 30-bit string 

011111110011001000111111100101 

ABRACADABRA ! All prefix-free codes are uniquely decodable (without needing any 
delimiters) in this way, so prefix-free codes are widely used in practice. Note that fixed- 
length codes such as 7-bit ASCII are prefix-free. 
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Trie representation for prefix-free codes. One conve¬ 
nient way to represent a prefix-free code is with a trie (see 
section 5 . 2 ). In fact, any trie with M null links defines a 
prefix-free code for M characters: we replace the null links 
by links to leaves (nodes with two null links), each con¬ 
taining a character to be encoded, and define the code for 
each character with the bitstring defined by the path from 
the root to the character, in the standard manner for tries 
where we associate 0 with moving left and 1 with moving 
right. For example, the figure at right shows two prefix- 
free codes for the characters in ABRACADABRA ! . On 
top is the variable-length code just considered; below is a 
code that produces the string 

11000111101011100110001111101 

which is 29 bits, 1 bit shorter. Is there a trie that leads to 
even more compression? How do we find the trie that 
leads to the best prefix-free code? It turns out that there is 
an elegant answer to these questions in the form of an al¬ 
gorithm that computes a trie which leads to a bitstream of 
minimal length for any given string. To make a fair com¬ 
parison with other codes, we also need to count the bits in 
the code itself, since the string cannot be decoded without 

it, and, as you will see, the code depends on the string. The general method for finding 
the optimal prefix-free code was discovered by D. Huffman (while a student!) in 1952 
and is called Huffman encoding. 

Overview. Using a prefix-free code for data compression involves five major steps. We 
view the bitstream to be encoded as a bytestream and use a prefix-free code for the 
characters as follows: 

■ Build an encoding trie. 

■ Write the trie (encoded as a bitstream) for use in expansion. 

■ Use the trie to encode the bytestream as a bitstream. 

Then expansion requires that we 

■ Read the trie (encoded at the beginning of the bitstream) 

■ Use the trie to decode the bitstream 

To help you best understand and appreciate the process, we consider these steps in 
order of difficulty. 


codeword table trie representation 



compressed bitstring 

011111110011001000111111100101 ->—30 bits 
A B RA CA DA B RA ! 


codeword table trie representation 



11000111101011100110001111101 —29 bits 

a b Fa Fa dab Fa I 

Two prefix-free codes 
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Trie nodes. We begin with the 
Node class at left. It is similar to 
the nested classes that we have 
used before to construct bi¬ 
nary trees and tries: each Node 
has left and right references 
to Nodes, which define the trie 
structure. Each Node also has 
an instance variable f req that 
is used in construction, and 
an instance variable ch, which 
is used in leaves to represent 
characters to be encoded. 

Expansion for prefix-free 
codes. Expanding a bitstream 
that was encoded with a prefix- 
free code is simple, given the 
trie that defines the code. The 
expand () method at left is an implementa¬ 
tion of this process. After reading the trie from 
standard input using the readTri e() method 
to be described later, we use it to expand the 
rest of the bitstream as follows: Starting at the 
root, proceed down the trie as directed by the 
bitstream (read in input bit, move left if it is 0, 
and move right if it is 1). When you encounter 
a leaf, output the character at that node and re¬ 
start at the root. If you study the operation of 
this method on the small prefix code example 
on the next page, you will understand and ap¬ 
preciate this process: For example, to decode 
the bitstring 011111001011. . .we start 
at the root, move left because the first bit is 0, 
output A; go back to the root, move right three times, then output B; go back to the root, 
move right twice, then left, then output R; and so forth. The simplicity of expansion is 
one reason for the popularity of prefix-free codes in general and Huffman compression 
in particular. 


public static void expand() 

{ 

Node root = readTrieO; 

int N = BinaryStdln.readlnt(); 

for (int i =0; i < N; i++) 

{ // Expand ith codeword. 

Node x = root; 
while (Ix.isLeafO) 

if (Bi naryStdln.readBoolean()) 
x = x. right; 
else x = x.left; 
BinaryStdOut.write(x.ch) ; 

} 

BinaryStdOut.closeQ ; 


Prefix-free code expansion (decoding) 


private static class Node implements Comparable<Node> 

{ // Huffman trie node 

private char ch; // unused for internal nodes 

private int freq; // unused for expand 

private final Node left, right; 

Node(char ch, int freq, Node left, Node right) 

{ 

this.ch = ch; 
this.freq = freq; 
this.left = left; 
this, right = right; 

} 

public boolean isl_eaf() 

{ return left == null && right == null; } 

public int compareTo(Node that) 

{ return this.freq - that.freq; } 


Trie node representation 
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private static String[] bui 1 dCode(Node root) 

{ // Make a lookup table from trie. 

String[] st = new String[R]; 
buildCode(st, root, 
return st; 

} 

private static void buildCode(String[] st, Node x, String s) 

{ // Make a lookup table from trie (recursive). 

if (x.isLeafO) 

{ st[x.ch] = s; return; } 
buildCode(st, x.left, s + '0'); 
buildCode(st, x.right, s + '!'); 


Building an encoding table from a (prefix-free) code trie 


Codeword table 


Trie representation 


Compression for prefix-free codes. For compression, we use 
the trie that defines the code to build the code table, as shown 
in the buildCodeO method at the top of this page. This 
method is compact and elegant, but a bit tricky, so it deserves 
careful study. For any trie, it produces a table giving the bit¬ 
string associated with each character in the trie (represented 
as a Stri ng of Os and Is). The coding table is a symbol table 
that associates a Stri ng with each character: we use a charac¬ 
ter-indexed array st[] instead of a general symbol table for 
efficiency, because the number of characters is not large. To create it, buildCodeO 
recursively walks the tree, maintaining a binary string that corresponds to the path 
from the root to each node (0 for left links and 1 for right links), and setting the code¬ 
word corresponding to each character when the character is found in a leaf. Once the 
coding table is built, compression is a simple matter: just look up the code for each 
character in the input. To use the encoding at right to compress ABRACADABRA! we 
write 0 (the codeword associated with A), 



then 111 (the codeword associated with 
B), then 110 (the codeword associated 
with R), and so forth. The code snippet at 
right accomplishes this task: we look up 
the Stri ng associated with each character 
in the input, convert it to 0/1 values in a 
char array, and write the corresponding 
bitstring to the output. 


for 

{ 


(int i =0; i < input.1ength; i++) 

String code = st[input[i]] ; 
for (int j = 0; j < code. 1 ength() ; j++) 
if (code.charAt(j) == '1') 

BinaryStdOut.write(true) ; 
else BinaryStdOut.write(false) ; 


Compression with an encoding table 
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Trie construction. For reference as we describe the process, the figure on the facing 
page illustrates the process of constructing a Huffman trie for the input 

it was the best of times it was the worst of times 

We keep the characters to be encoded in leaves and maintain the f req instance variable 
in each node that represents the frequency of occurrence of all characters in the subtree 
rooted at that node. The first step is to create a forest of 1-node trees (leaves), one for 
each character in the input stream, each assigned a f req value equal to its frequency 
of occurrence in the input. In the example, the input has 8 ts, 5 es, 11 spaces, and so 
forth. ( Important note: To obtain these frequencies, we need to read the whole input 
stream—Huffman encoding is a two-pass algorithm because we will need to read the 
input stream a second time to compress it.) Next, we build the coding trie from the 
bottom up according to the frequencies. When building the trie, we view it as a binary 
trie with frequencies stored in the nodes; after it has been built, we view it as a trie for 
coding, as just described. The process works as follows: we find the two nodes with the 
smallest frequencies and then create a new node with those two nodes as children (and 
with frequency value set to the sum of the values of the children). This operation re¬ 
duces the number of tries in the forest by one. Then we iterate the process: find the two 
nodes with smallest frequency in that forest and a create a new node created in the same 
way. Implementing the process is straightforward with a priority queue, as shown in the 
bui 1 dT ri e () method on page 830. (For clarity, the tries in the figure are kept in sorted 
order.) Continuing, we build up larger and larger tries and at the same time reduce the 
number of tries in the forest by one at each step (remove two, add one). Ultimately, all the 

private static Node buildTrie(int[] freq) 

{ 

// Initialize priority queue with singleton trees. 

MinPQ<Node> pq = new MinPQ<Node>(); 

for (char c = 0; c < R; C++) 
if (freq[c] > 0) 

pq.insert(new Node(c, freq[c], null, null)); 

while (pq.sizeO > 1) 

{ // Merge two smallest trees. 

Node x = pq.delMinO; 

Node y = pq.delMinO; 

Node parent = new Node('\0', x.freq + y.freq, x, y); 
pq . insert(parent); 

} 

return pq.delMinO; 

} 

Building a Huffman encoding trie 
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trie representation 


codeword table 



key 

value 

LF 

101010 

SP 

01 

a 

non 

b 

101011 

e 

000 

f 

11000 

h 

11001 

i 

1011 

m 

11010 

0 

0011 

r 

10100 

s 

100 

t 

111 

w 

0010 


Huffman code for the character stream “it was the best of times it was the worst of times ” 


nodes are combined together into a single trie. The leaves in this trie have the characters 
to be encoded and their frequencies in the input; each non-leaf node is the sum of the 
frequencies of its two children. Nodes with low frequencies end up far down in the trie, 
and nodes with high frequencies end up near the root of the trie. The frequency in the 
root equals the number of characters in the input. Since it is a binary trie with charac¬ 
ters only in its leaves, it defines a prefix-free code for the characters. Using the codeword 
table created by bui 1 dCode() for this example (shown at right in the diagram at the 
top of this page), we get the output bitstring 

10111110100101101110001111110010000110101100 - 

01001110100111100001111101111010000100011011 - 

11101001011011100011111100100001001000111010 - 

01001110100111100001111101111010000100101010 . 

which is 176 bits, a savings of 57 percent over the 408 bits needed to encode the 51 
characters in standard 8-bit ASCII (not counting the cost of including the code, which 
we will soon consider). Moreover, since it is a Huffman code, no other prefix-free code 
can encode the input with fewer bits. 

Optimality. We have observed that high-frequency characters are nearer the root of 
the tree than lower-frequency characters and are therefore encoded with fewer bits, so 
this is a good code, but why is it an optimal prefix-free code? To answer this question, we 
begin by defining the weighted external path length of a tree to be the sum of the weight 
(associated frequency count) times depth (see page 226) of all of the leaves. 
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Proposition T. For any prefix-free code, the length of the encoded bitstring is equal 
to the weighted external path length of the corresponding trie. 

Proof: The depth of each leaf is the number of bits used to encode the character in 
the leaf. Thus, the weighted external path length is the length of encoded bitstring: 
it is equivalent to the sum over all letters of the number of occurrences times the 
number of bits per occurrence. 


For our example, there is one leaf at distance 2 (sp, with frequency 11), three leaves at 
distance 3 (e, s, and t, with total frequency 19), three leaves at distance 4 (w, o, and i, 
with total frequency 10), five leaves at distance 5 (r, f, h, m, and a, with total frequency 
9) and two leaves at distance 6 (lf and b, with total frequency 2), so the sum total is 2T1 
3T9 + 4T0 + 5-9 + 6-2 =176, the length of the output bitstring, as expected. 


Proposition U. Given a set of r symbols and frequencies, the Huffman algorithm 
builds an optimal prefix-free code. 

Proof: By induction on r. Assume that the Huffman code is optimal for any set 
of fewer than r symbols. Let T H be the code computed by Huffman for the set of 
symbols and associated frequencies (s |; r t ),..., (s r ,/ r ) and denote the length of the 
code (weighted external path length of the trie) by W{T H ). Suppose that (s ; ,/ ; ) and 
are the first two symbols chosen. The algorithm then computes the code T H * 
for the set of n~ 1 symbols with (s ; ,/ ; ) and (s,-,/j) replaced by (+ ) where s* is 
a new symbol in a leaf at some depth d. Note that 

W(T H ) = W(T H *) - d{Z+fy + (d + Dif+fj) = W(T H *) + tf.+jj.) 

Now consider an optimal trie T for (s,, r,),..., ( s r ,f r ), of height h. Note that that 
(s ; ,/ ; ) and (Spfj) must be at depth h (else we could make a trie with lower external 
path length by swapping them with nodes at depth h). Also, assume {s t , f) and 
are siblings by swapping (s-, f) with {s i ,fY s sibling. Now consider the tree T* 
obtained by replacing their parent with (s*,/) +/■). Note that (by the same argument 
as above) W{T) = W{T*) + 

By the inductive hypothesis T H * is optimal: W{T H *) £ W(T*). Therefore, 

W(T h ) = W(T h *) + (f+fj) < W(T*) + (f + fj) = W(T) 

Since T is optimal, equality must hold, and T H is optimal. 
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Whenever a node is to be picked, it can be the case that there are several nodes with 
the same weight. Huffman’s method does not specify how such ties are to be broken. It 
also does not specify the left/right positions of the children. Different choices lead to 

different Huffman codes, but all such 
codes will encode the message with 
the optimal number of bits among 
prefix-free codes. 

Writing and reading the trie. As we 
have emphasized, the savings figure 
quoted above is not entirely accurate, 
because the compressed bitstream 
cannot be encoded without the trie, 
so we must account for the cost of 
including the trie in the compressed 
output, along with the bitstring. For 
long inputs, this cost is relatively 
small, but in order for us to have a 
full data-compression scheme, we must write the trie onto a bitstream when compress¬ 
ing and read it back when expanding. How can we encode a trie as a bitstream, and then 
expand it? Remarkably, both tasks can be achieved with simple recursive procedures, 
based on a preorder traversal of the trie. The procedure wr i teTr i e 0 below traverses a 
trie in preorder: when it visits an internal node, it writes a single 0 bit; when it visits a 
leaf, it writes a 1 bit, followed by the 8-bit ASCII code of the character in the leaf. The 
bitstring encoding of the Huffman trie for our ABRACADABRA ! example is shown 
above. The first bit is 0, corresponding to the root; since the leaf containing A is encoun¬ 
tered next, the next bit is 1, followed by 
0100001, the 8-bit ASCII code for A; 
the next two bits are 0 because two in¬ 
ternal nodes are encountered next, and 
so forth. The corresponding method 
readTrieO on page 835 reconstructs 
the trie from the bitstring: it reads 
a single bit to learn which type of 
node comes next: if a leaf (the bit is 
1 ) it reads the next character and cre¬ 
ates a leaf; if an internal node (the bit 
is 0) it creates an internal node and 


private static void writeTrie(Node x) 

{ // Write bitstring-encoded trie, 

if (x.isLeaf()) 

{ 

BinaryStdOut.write(true) ; 
BinaryStdOut.write(x.ch); 
return; 

} 

BinaryStdOut.write(false) ; 
writeTrie(x.left) ; 
writeTrie(x. right); 


Writing a trie as a bitstring 


leaves 

\ A 



01010000010010100010001000010101010000110101010010101000010 
1 tt t 1 

1234 5 - internal nodes 

Using preorder traversal to encode a trie as a bitstream 
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private static Node readTrieO 
{ 

if (Bi naryStdln. readBooleanO) 

return new Node(BinaryStdln.readChar(), 0, null, null); 
return new Node('\0', 0, readTrieO, readTrieO); 

} 

Reconstructing a trie from the preorder bitstring representation 

then (recursively) builds its left and right subtrees. Be sure that you understand these 
methods: their simplicity is somewhat deceiving. 

Huffman compression implementation. Along with the methods buildCodeO, 
buildTrieO, readTrieO and writeTrieO that we have just considered (and 
the expand () method that we considered first), algorithm 5.10 is a complete imple¬ 
mentation of Huffman compression. To expand the overview that we considered sev¬ 
eral pages earlier, we view the bitstream to be encoded as a stream of 8-bit char values 
and compress it as follows: 

■ Read the input. 

■ Tabulate the frequency of occurrence of each char value in the input. 

■ Build the Huffman encoding trie corresponding to those frequencies. 

■ Build the corresponding codeword table, to associate a bitstring with each char 
value in the input. 

■ Write the trie, encoded as a bitstring. 

■ Write the count of characters in the input, encoded as a bitstring. 

■ Use the codeword table to write the codeword for each input character. 

To expand a bitstream encoded in this way, we 

■ Read the trie (encoded at the beginning of the bitstream) 

■ Read the count of characters to be decoded 

■ Use the trie to decode the bitstream 

With four recursive trie-processing methods and a seven-step compression process, 
Huffman compression is one of the more involved algorithms that we have considered, 
but it is also one of the most widel-used, because of its effectiveness. 
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ALGORITHM 5.10 Huffman compression 


public class Huffman 

{ 

private static int R = 256; // ASCII alphabet 

// See page 828 for inner Node class. 

// See text for helper methods and expandf). 

public static void compressO 

{ 

// Read input. 

String s = BinaryStdln.readString() ; 
char[] input = s.toCharArrayO ; 

// Tabulate freguency counts. 

int[] freq = new int[R]; 
for (int i = 0; i < input.1ength; i++) 
freq[input[i]]++; 

// Build Huffman code trie. 

Node root = bui1dTrie(freq); 

// Build code table (recursive). 

String[] st = new StringfR]; 
buildCode(st, root, ""); 

// Print trie for decoder (recursive). 
writeTrie(root); 

// Print number of chars. 

BinaryStdOut.write(input.1ength) ; 

// Use Huffman code to encode input, 
for (int i = 0; i < input.1ength; i++) 

{ 

String code = st[input[i]]; 

for (int j =0; j < code.1ength(); j++) 

if (code.charAt(j) == '1') 

BinaryStdOut.write(true); 
else BinaryStdOut.write(false); 

} 

BinaryStdOut.closeO ; 

} 

} 


This implementation of Huffman encoding builds an explicit coding trie, using various helper meth¬ 
ods that are presented and explained in the last several pages of text. 
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test case (96 bits) 

% more abra.txt 
ABRACADABRA! 

% java Huffman - < abra.txt | java BinaryDump 60 
010100000100101000100010000101010100001101010100101010000100 
000000000000000000000000000110001111100101101000111110010100 

120 bi tS -« - compression ratio 120/96 = 125% due to 59 bits for trie and 32 bits for count 


example from text (408 bits) 

% more tinytinyTale.txt 

it was the best of times it was the worst of times 

% java Huffman - < tinytinyTale.txt | java BinaryDump 64 
0001011001010101110111101101111100100000001011100110010111001001 
0000101010110001010110100100010110011010110100001011011011011000 
0110111010000000000000000000000000000110011101111101001011011100 
0111111001000011010110001001110100111100001111101111010000100011 
0111110100101101110001111110010000100100011101001001110100111100 
00111110111101000010010101000000 

352 bi tS -* - compression ratio 352/408 = 86% even with 137 bits for trie and 32 bits for count 

% java Huffman - < tinytinyTale.txt | java Huffman + 
it was the best of times it was the worst of times 

first chapter of Tale of Two Cities 


% java PictureDump 512 90 < medTale.txt 



45056 bits 


% java Huffman - < medTale.txt | java PictureDump 512 47 



23912 bits -« - compression ratio 23912/45056 = 53% 


entire text of Tale of Two Cities 

% java BinaryDump 0 < tale.txt 
5812552 bits 

% java Huffman - < tale.txt > tale.txt.huf 

% java BinaryDump 0 < tale.txt.huf 

3043928 bits -* - compression ratio 3043928/5812552 = 52% 


Compressing and expanding bytestreams with Huffman encoding 
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one reason for the popularity of Huffman compression is that it is effective for 
various types of files, not just natural language text. We have been careful to code the 
method so that it can work properly for any 8-bit value in each 8-bit character. In 
other words, we can apply it to any bytestream whatsoever. Several examples, for file 
types that we have considered earlier in this section, are shown in the figure at the bot¬ 
tom of this page. These examples show that Huffman compression is competitive with 
both fixed-length encoding and run-length encoding, even though those methods are 
designed to perform well for certain types of files. Understanding the reason Huff¬ 
man encoding performs well in these domains is instructive. In the case of genomic 
data, Huffman compression essentially discovers a 2-bit code, as the four letters appear 
with approximately equal frequency so that the Huffman trie is balanced, with each 
character assigned a 2-bit code. In the case of run-length encoding, 00000000 and 
11111111 are likely to be the most frequently occurring characters, so they are likely 
to be encoded with 2 or 3 bits, leading to substantial compression. 


virus (50000 bits) 


% java Genome - < genomeVirus.txt | java PictureDump 512 25 

12556 

% java Huffman - < genomeVirus.txt | java PictureDump 512 25 

12576 bits •* - Huffman compression needs just 40 more bits than custom 2-bit code 

bitmap (1536 bits) 

% java RunLength - < q32x48.bin | java BinaryDump 0 
1144 bits 

% java Huffman - < q32x48.bin | java BinaryDump 0 

816 bi tS — - Huffman compression uses 29% fewer bits than customized method 

higher-resolution bitmap (6144 bits) 

% java RunLength - < q64x96.bin | java BinaryDump 0 
2296 bits 


% java Huffman - < q64x96.bin | java BinaryDump 0 
2032 bits -* - gap narrows to 11 % for higher resolution 


Compressing and expanding genomic data and bitmaps with Huffman encoding 
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A remarkable alternative to Huffman compression that was developed in the late 
1970s and the early 1980s by A. Lempel, J. Ziv, and T. Welch has emerged as one of the 
most widely used compression methods because it is easy to implement and works well 
for a variety of file types. 

The basic plan complements the basic plan for Huffman coding. Rather than main¬ 
tain a table of variable- length codewords for fixed-length patterns in the input, we 
maintain a table of /bced-length codewords for variable- length patterns in the input. A 
surprising added feature of the method is that, by contrast with Huffman encoding, we 
do not have to encode the table. 

LZW compression. To fix ideas, we will consider a compression example where we 
read the input as a stream of 7-bit ASCII characters and write the output as a stream of 
8-bit bytes. (In practice, we typically use larger values for these parameters—our imple¬ 
mentations use 8-bit inputs and 12-bit outputs.) We refer to input bytes as characters, 
sequences of input bytes as strings, and output bytes as codewords, even though these 
terms have slightly different meanings in other contexts. The TZW compression algo¬ 
rithm is based on maintaining a symbol table that associates string keys with (fixed- 
length) codeword values. We initialize the symbol table with the 128 possible single¬ 
character string keys and associate them with 8-bit codewords obtained by prepending 
0 to the 7-bit value defining each character. For economy and clarity, we use hexadeci¬ 
mal to refer to codeword values, so 41 is the codeword for ASCII A, 52 for R, and so 
forth. We reserve the codeword 80 to signify end of file. We will assign the rest of the 
codeword values (81 through FF) to various substrings of the input that we encounter, 
by starting at 81 and incrementing the value for each new key added. To compress, we 
perform the following steps as long as there are unscanned input characters: 

■ Find the longest string s in the symbol table that is a prefix of the unscanned 
input. 

■ Write the 8-bit value (codeword) associated with s. 

■ Scan one character past s in the input. 

■ Associate the next codeword value with s + c (c appended to s) in the symbol 
table, where c is the next character in the input. 

In the last of these steps, we look ahead to see the next character in the input to build 
the next dictionary entry, so we refer to that character c as the lookahead character. 
For the moment, we simply stop adding entries to the symbol table when we run out 
of codeword values (after assigning the value FF to some string)—we will later discuss 
alternate strategies. 
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LZW compression example. The figure below gives details of the operation of LZW 
compression for the example input ABRACADABRABRABRA. For the first seven 
characters, the longest prefix match is just one character, so we output the codeword 
associated with the character and associate the codewords from 81 through 87 to two- 
character strings. Then we find prefix matches with AB (so we output 81 and add ABR to 
the table), RA (so we output 83 and add RAC to the table), BR (so we output 82 and add 
BRA to the table), and ABR (so we output 88 and add ABRA to the table), leaving the last 
A (so we output its codeword, 41). 


input A 

B R 

A 

c 

A 

D 

A B 

R A 

B R 

A B 

R A 

EOF 

matches A 

B R 

A 

c 

A 

D 

A B 

R A 

B R 

ABR 

A 

\ 

output 41 

42 52 

41 

43 

41 

44 

81 

83 

82 

88 

41 

80 
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RAB 
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ABRA 8B 
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8B 


LZW compression for ABRACADABRABRABRA 


The input is 17 ASCII characters of 7 bits each for a total of 119 bits; the output is 12 
codewords of 8 bits each for a total of 96 bits—a compression ratio of 82 percent even 
for this tiny example. 

LZW trie representation. LZW compression involves two symbol-table operations: 

■ Find a longest-prefix match of the input with a symbol-table key. 

■ Add an entry associating the next code¬ 
word with the key formed by appending 
the lookahead character to that key. 

Our trie data structures of section 5.2 are tai¬ 
lor-made for these operations. The trie repre¬ 
sentation for our example is shown at right. To 
find a longest prefix match, we traverse the trie 
from the root, matching node labels with input 
characters; to add a new codeword, we connect 
a new node labeled with the next codeword and 
the lookahead character to the node where the 
search terminated. In practice, we use a TST for 
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space efficiency, as described in section 5 . 2 . The contrast with the use of tries in Huff¬ 
man encoding is worth noting: for Huffman encoding, tries are useful because no prefix 
of a codeword is also a codeword; for LZW tries are useful because every prefix of an 
input-substring key is also a key. 

LZW expansion. The input for TZW expansion in our example is a sequence of 8 -bit 
codewords; the output is a string of 7-bit ASCII characters. To implement expansion, 
we maintain a symbol table that associates strings of characters with codeword values 
(the inverse of the table used for compression). We fill the table entries from 00 to 7F 
with one-character strings, one for each ASCII character, set the first unassigned code¬ 
word value to 81 (reserving 80 for end of file), set the current string val to the one- 
character string consisting of the first character, and perform the following steps until 
reading codeword 80 (end of file): 

■ Write the current string val. 

■ Read a codeword x from the input. 

■ Set s to the value associated with x in the symbol table. 

■ Associate the next unasssigned codeword value to val + c in the symbol table, 
where c is the first character of s. 

■ Set the current string val to s. 

This process is more complicated than compression because of the lookahead charac¬ 
ter: we need to read the next codeword to get the first character in the string associated 
with it, which puts the process one step out of synch. For the first seven codewords, we 
just look up and write the appropriate character, then look ahead one character and 
add a two-character entry to the symbol table, as before. Then we read 81 (so we write 
A B and add A B R to the table), 83 (so we write R A and add R A B to the table), 82 (so 
we write B R and add B R A to the table), and 88 (so we write A B R and add A B R A to 
the table), leaving 41. Finally we read the end-of-file character 80 (so we write A). At 
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LZW expansion for 41 42 52 41 43 41 44 81 83 82 88 41 80 


www.it-ebooks.info 


842 


CHAPTERS Strings 


ALG0RITHM5.il LZW compression 


public class LZW 

{ 

private static final int R = 256; // number of input chars 

private static final int L = 4096; // number of codewords = 2A12 

private static final int W = 12; // codeword width 

public static void compressO 

{ 

String input = BinaryStdln.readString(); 

TST<Integer> st = new TST<Integer>(); 

for (int i =0; i < R; i++) 
st.putC"" + (char) i, i); 
int code = R+l; // R is codeword for EOF. 


while (input.length() > 0) 

{ 

String s = st.longestPrefixOf(input); 
Bi naryStdOut.write(st.get(s), W); 
int t = s.lengthO ; 
if (t < input.lengthO && code < L) 
st.put(input.substring(0, t + 1), 
input = input.substring(t); 

} 


// Find max prefix match. 
// Print s's encoding. 

// Add s to symbol table. 
code++); 

// Scan past s in input. 


BinaryStdOut.write(R, W); // Write EOF. 

BinaryStdOut.closeQ ; 


public static void expandO 

// See page 844. 


This implementation of Lempel-Ziv-Welch data compression uses 8-bit input bytes and 12-bit code¬ 
words and is appropriate for arbitrary large files. Its codewords for the small example are similar to 
those discussed in the text: the single-character codewords have a leading 0; the others start at 100. 


% more abraLZW.txt 
ABRACADABRABRABRA 

% java LZW - < abraLZW.txt | java HexDump 20 

04 10 42 05 20 41 04 30 41 04 41 01 10 31 02 10 80 41 10 00 

160 bits 
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the end of the process, we have written the original input, as expected, and also built 
the same code table as for compression, but with the key-value roles inverted. Note that 
we can use a simple array-of-strings representation for the table, indexed by codeword. 

Tricky situation. There is a subtle bug in the process just described, one that is of¬ 
ten discovered by students (and experienced programmers!) only after developing an 
implementation based on the description above. 

The problem, illustrated in the example at right, is 
that it is possible for the lookahead process to get 
one character ahead of itself. In the example, the 
input string 

ABABABA 

is compressed to five output codewords 
41 42 81 83 80 

as shown in the top part of the figure. To expand, 
we read the codeword 41, output A, read the code¬ 
word 42 to get the lookahead character, add AB as 
table entry 81, output the B associated with 42, 
read the codeword 81 to get the lookahead char¬ 
acter, add BA as table entry 82, and output the AB 
associated with 81. So far, so good. But when we 
read the codeword 83 to get the lookahead character, we are stuck, because the reason 
that we are reading that codeword is to complete table entry 83! Fortunately, it is easy 
to test for that condition (it happens precisely when the codeword is the same as the 
table entry to be completed) and to correct it (the lookahead character must be the first 
character in that table entry, since that will be the next character to be output). In this 
example, this logic tells us that the lookahead character must be A (the first character in 
ABA). Thus, both the next output string and table entry 83 should be ABA. 

Implementation. With these descriptions, implementing LZW encoding is straight¬ 
forward, given in algorithm 5.11 on the facing page (the implementation of expand 0 
is on the next page). These implementations take 8 -bit bytes as input (so we can com¬ 
press any file, not just strings) and produce 12 -bit codewords as output (so that we can 
get better compression by having a much larger dictionary). These values are specified 
in the (final) instance variables R, L, and W in the code. We use a TST (see section 5 . 2 ) 
for the code table in compress O (taking advantage of the ability of trie data structures 
to support efficient implementations of longestPrefixOfQ) and an array of strings 


compression 

input A 

B 

A 

BABA 

matches A 

B 

A B 

ABA 

output 41 

42 

81 

83 80 

codeword table 

key value 

AB 81 

AB 

AB 81 


BA 82 BA 
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83 ABA 83 
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81 

83 80 

output A 
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A B 

7 ^ must be ABA 

(see below) 

81 A B 

AB 

AB 



82 BA 

BA 

83 AB? 

to complete entry 


t 

next character in output — the lookahead character! 


LZW expansion: tricky situation 
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ALG0RITHM5.il (continued) LZW expansion 


public static void expand0 

{ 

String[] st = new String[L]; 

int i; // next available codeword value 

for (i =0; i < R; i++) // Initialize table for chars. 

st[i] = "" + (char) i; 

st[i++] = " // (unused) lookahead for EOF 


int codeword = BinaryStdln.readlnt(W); 
String val = st[codeword]; 
while (true) 

{ 


BinaryStdOut.write(val); // Write current substring, 

codeword = BinaryStdln.readlnt(W); 


if (codeword == R) break; 

String s = st[codeword]; 
if (i == codeword) 

s = val + val.charAt(O); 
if (i < L) 

st[i++] = val + s.charAt(O); 
val = s; 


// Get next codeword. 

// If lookahead is invalid, 

// make codeword from last one. 

// Add new entry to code table. 

// Update current codeword. 


BinaryStdOut.closeO; 

} 


This implementation of expansion for the Lempel-Ziv-Welch algorithm is a bit more complicated 
than compression because of the need to extract the lookahead character from the next codeword 
and because of a tricky situation where lookahead is invalid (see text). 


% java LZW - < abraLZW.txt | java LZW + 
ABRACADABRABRABRA 

% more ababLZW.txt 
ABABABA 

% java LZW - < ababLZW.txt | java LZW + 
ABABABA 
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for the inverse code table in expand(). With these choices, the code for compressO 
and expand () is little more than a line-by-line translation of the descriptions in the 
text. These methods are very effective as they stand. For certain files, they can be further 
improved by emptying the codeword table and starting over each time all the codeword 
values are used. These improvements, along with experiments to evaluate their effec¬ 
tiveness, are addressed in the exercises at the end of this section. 

as usual, it is worth your while to study carefully the examples given with the pro¬ 
grams and at the bottom of this page of LZW compression in action. Over the several 
decades since its invention, it has proven to be a versatile and effective data-compres- 
sion method. 


virus (50000 bits) 


% java Genome - < genomeVirus.txt | java PictureDump 512 25 



12536 bits 


% java LZW - < genomeVirus.txt | java PictureDump 512 36 



18232 bits -*- not as good as 2-bit code because repetitive data is rare 


bitmap (6144 bits) 

% java RunLength - < q64x96.bin | java BinaryDump 0 
2296 bits 

% java LZW - < q64x96.bin | java BinaryDump 0 
2824 bi ts - not as good as run-length code because file size is too small 


entire text of Tale of Two Cities (5812552 bits) 

% java BinaryDump 0 < tale.txt 
5812552 bits 

% java Huffman - < tale.txt | java BinaryDump 0 
3043928 bits 

% java LZW - < tale.txt | java BinaryDump 0 
2667952 bi ts -<- compression ratio 2667952/5812552 = 46 % (bestyet) 

Compressing and expanding various files with LZW 12-bit encoding 
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| Q&A 

Q. Why Bi naryStdln and Bi naryStdOut? 

A. It’s a tradeoff between efficiency and convenience. Stdln can handle 8 bits at a time; 
Bi naryStdln has to handle each bit. Most applications are bytestream-oriented; data 
compression is a special case. 

Q. Why close() ? 

A. This requirement stems from the fact that standard output is actually a bytestream, 
so Bi naryStdOut needs to know when to write the last byte. 

Q. Can we mix Stdln and Bi naryStdln ? 

A. That is not a good idea. Because of system and implementation dependencies, there 
is no guarantee of what might happen. Our implementations will raise an exception. 
On the other hand, there is no problem with mixing StdOut and Bi naryStdOut (we do 
it in our code). 

Q. Why is the Node class stati c in Huffman? 

A. Our data-compression algorithms are organized as collections of static methods, 
not data-type implementations. 

Q. Can I at least guarantee that my compression algorithm will not increase the length 
of a bitstream? 

A. You can just copy it from input to output, but you still need to signify not to use 
a standard compression scheme. Commercial implementations sometimes make this 
guarantee, but it is quite weak and far from universal compression. Indeed, typical 
compression algorithms do not even make it past the second step of our first proof of 
proposition s: few algorithms will further compress a bitstring produced by that same 
algorithm. 
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EXERCISES 


5 . 5.1 Consider the four variable-length codes shown 
in the table at right. Which of the codes are prefix-free? 
Uniquely decodable? For those that are uniquely decod- 

symbol 

code 1 

code 2 

code 3 

code 4 

A 

0 

0 

1 

1 

able, give the encoding of 1000000000000. 

B 

100 

1 

01 

01 

5 . 5.2 Given a example of a uniquely decodable code 

C 

10 

00 

001 

001 

that is not prefix-free. 

D 

11 

11 

0001 

000 


Answer: Any suffix-free code is uniquely decodable. 

5 . 5.3 Give an example of a uniquely decodable code that is not prefix free or suffix free. 
Answer: { 0011, Oil, 11, 1110} or {01, 10, Oil, 110} 

5 . 5.4 Are {01, 1001, 1011, 111, 1110}and {01, 1001, 1011, 111, 1110 } unique¬ 
ly decodable? If not, find a string with two encodings. 

5 . 5.5 Use RunLength onthefileql28xl92.bin from the booksite. How many bits are 
there in the compressed file? 

5 . 5.6 How many bits are needed to encode N copies of the symbol a (as a function 
of N)? N copies of the sequence abc? 

5 . 5.7 Give the result of encoding the strings a, aa, aaa, aaaa,... (strings consisting of 
N as) with run-length, Huffman, and LZW encoding. What is the compression ratio 
as a function of N? 

5 . 5.8 Give the result of encoding the strings ab, abab, ababab, abababab, ... (strings 
consisting of N repetitions of ab) with run-length, Huffman, and LZW encoding. What 
is the compression ratio as a function of N? 

5 . 5.9 Estimate the compression ratio achieved by run-length, Huffman, and LZW en¬ 
coding for a random ASCII string of length N (all characters equally likely at each posi¬ 
tion, independently). 

5 . 5.10 In the style of the figure in the text, show the Huffman coding tree construction 
process when you use Huffman for the string "it was the age of foolishness”. 
How many bits does the compressed bitstream require? 
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EXERCISES (continued) 


5.5.11 What is the Huffman code for a string whose characters are all from a two- 
character alphabet? Give an example showing the maximum number of bits that could 
be used in a Huffman code for an N-c haracter string whose characters are all from a 
two-character alphabet. 

5.5.12 Suppose that all of the symbol probabilities are negative powers of 2. Describe 
the Huffman code. 

5.5.13 Suppose that all of the symbol frequencies are equal. Describe the Huffman code. 

5.5.14 Suppose that the frequencies of the occurrence of all the characters to be encoded 
are different. Is the Huffman encoding tree unique? 

5.5.15 Huffman coding could be extended in a straightforward way to encode in 2-bit 
characters (using 4-way trees). What would be the main advantage and the main dis¬ 
advantage of doing so? 

5.5.16 What is the LZW encoding of the following inputs? 

a. TOBEORNOTTOBE 

b. YABBADABBADABBADOO 

c. AAAAAAAAAAAAAAAAAAAAA 

5.5.17 Characterize the tricky situation in LZW coding. 

Solution : Whenever it encounters cScSc, where c is a symbol and S is a string, cS is in 
the dictionary already but cSc is not. 

5.5.18 Let F k be the k th Fibonacci number. Consider N symbols, where the k th symbol 
has frequency F k . Note that F k + F 1 + ... + F N = F N+2 — 1. Describe the Huffman code. 
Flint: The longest codeword has length N — 1. 

5.5.19 Show that there are at least 2 N_1 different Huffman codes corresponding to a 
given set of N symbols. 

5.5.20 Give a Huffman code where the frequency of Os in the output is much, much 
higher than the frequency of Is. 

Answer : If the character A occurs 1 million times and the character B occurs once, the 
codeword for A will be 0 and the codeword for B will be 1. 
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5 . 5.21 Prove that the two longest codewords in a Huffman code have the same length. 

5 . 5.22 Prove the following fact about Huffman codes: If the frequency of symbol i is 
strictly larger than the frequency of symbol;, then the length of the codeword for sym¬ 
bol i is less than or equal to the length of the codeword for symbol;. 

5 . 5.23 What would be the result of breaking up a Huffman-encoded string into five-bit 
characters and Huffman-encoding that string? 

5 . 5.24 In the style of the figures in the text, show the encoding trie and the compression 
and expansion processes when LZW is used for the string 

it was the best of times it was the worst of times 
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CREATIVE PROBLEMS 

5 . 5.25 Fixed length width code. Implement a class RLE that uses fixed-length encod¬ 
ing, to compress ASCII bytestreams using relatively few different characters, including 
the code as part of the encoded bitstream. Add code to compress () to make a string 
al pha with all the distinct characters in the message and use it to make an A1 phabet for 
use in compressO, prepend al pha ( 8 -bit encoding plus its length) to the compressed 
bitstream, then add code to expand () to read the alphabet before expansion. 

5 . 5.26 Rebuilding the LZW dictionary. Modify LZW to empty the dictionary and start 
over when it is full. This approach is recommended in some applications because it bet¬ 
ter adapts to changes in the general character of the input. 

5 . 5.27 Long repeats. Estimate the compression ratio achieved by run-length, Huff¬ 
man, and LZW encoding for a string of length 2 N formed by concatenating two copies 
of a random ASCII string of length N (see exercise 5 . 5 . 9 ), under any assumptions that 
you think are reasonable. 
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llll Context 



C omputing devices are ubiquitous in the modern world. In the last several 
decades, we have evolved from a world where computing devices were virtually 
unknown to a world where billions of people use them regularly. Moreover, to¬ 
day’s cellphones are orders of magnitude more powerful than the supercomputers that 
were available only to the privileged few as little as 30 years ago. But many of the under¬ 
lying algorithms that enable these devices to work effectively are the same ones that we 
have studied in this book. Why? Survival of the fittest. Scalable (linear and linearithmic) 
algorithms have played a central role in the process and validate the idea that efficient 
algorithms are important. Researchers of the 1960s and 1970s built the basic infrastruc¬ 
ture that we now enjoy with such algorithms. They knew that scalable algorithms are 
the key to the future; the developments of the past several decades have validated that 
vision. Now that the infrastructure is built, people are beginning to use it, for all sorts 
of purposes. As B. Chazelle has famously observed, the 20th century was the century of 
the equation, but the 21st century is the century of the algorithm. 

Our treatment of fundamental algorithms in this book is only a starting point. The 
day is soon coming (if it is not already here) when one could build a college major 
around the study of algorithms. In commerical applications, scientific computing, en¬ 
gineering, operations research (OR), and countless other areas of inquiry too diverse 
to even mention, efficient algorithms make the difference between being able to solve 
problems in the modern world and not being able to address them at all. Our empha¬ 
sis throughout this book has been to study important and useful algorithms. In this 
chapter, we reinforce this orientation by considering examples that illustrate the role of 
the algorithms that we have studied (and our approach to the study of algorithms) in 
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several advanced contexts. To indicate the scope of the impact of the algorithms, we be¬ 
gin with a very brief description of several important areas of application. To indicate 
the depth, we later consider specific representative examples in detail and introduce the 
theory of algorithms. In both cases, this brief treatment at the end of a long book can 
only be indicative, not inclusive. For every area of application that we mention, there 
are dozens of others, equally broad in scope; for every point that we describe within 
an application, there are scores of others, equally important; and for every detailed 
example we consider, there are hundreds if not thousands of others, equally impactful. 

Commercial applications. The emergence of the internet has underscored the central 
role of algorithms in commercial applications. All of the applications that you use regu¬ 
larly benefit from the classic algorithms that we have studied: 

■ Infrastructure (operating systems, databases, communications) 

■ Applications (email, document processing, digital photography) 

■ Publishing (books, magazines, web content) 

■ Networks (wireless networks, social networks, the internet) 

■ Transaction processing (financial, retail, web search) 

As a prominent example, we consider in this chapter B-trees, a venerable data structure 
that was developed for mainstream computers of the 1960s but still serve as the basis 
for modern database systems. We will also discuss suffix arrays, for text indexing. 

Scientific computing. Since von Neumann developed mergesort in 1950, algorithms 
have played a central role in scientific computing. Today’s scientists are awash in ex¬ 
perimental data and are using both mathematical and computational models to under¬ 
stand the natural world for: 

■ Mathematical calculations (polynomials, matrices, differential equations) 

■ Data processing (experimental results and observations, especially genomics) 

■ Computational models and simulation 

All of these can require complex and extensive computing with huge amounts of data. 
As a detailed example of an application in scientific computing, we consider in this 
chapter a classic example of event-driven simulation. The idea is to maintain a model of 
a complicated real-world system, controlling changes in the model over time. There are 
a vast number of applications of this basic approach. We also consider a fundamental 
data-processing problem in computational genomics. 

Engineering. Almost by definition, modern engineering is based on technology. Mod¬ 
ern technology is computer-based, so algorithms play a central role for 

■ Mathematical calculations and data processing 

■ Computer-aided design and manufacturing 
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■ Algorithm-based engineering (networks, control systems) 

■ Imaging and other medical systems 

Engineers and scientists use many of the same tools and approaches. For example, sci¬ 
entists develop computational models and simulations for the purpose of understand¬ 
ing the natural world; engineers develop computational models and simulations for the 
purpose of designing, building, and controlling they artifacts they create. 

Operations research. Researchers and practitioners in OR develop and apply math¬ 
ematical models for problem solving, including 

■ Scheduling 

■ Decision making 

■ Assignment of resources 

The shortest-paths problem of section 4.4 is a classic OR problem. We revisit this 
problem and consider the maxflow problem, illustrate the importance of reduction, and 
discuss implications for general problem-solving models, in particular the linear pro¬ 
gramming model that is central in OR. 

algorithms play an important role in numerous subfields of computer science 
with applications in all of these areas, including, but certainly not limited to 

■ Computational geometry 

■ Cryptography 

■ Databases 

■ Programming languages and systems 

■ Artificial intelligence 

In each field, articulating problems and finding efficient algorithms and data structures 
for solving them play an essential role. Some of the algorithms we have studied apply 
directly; more important, the general approach of designing, implementing, and ana¬ 
lyzing algorithms that lies at the core of this book has proven successful in all of these 
fields. This effect is spreading beyond computer science to many other areas of inquiry, 
from games to music to linguistics to finance to neuroscience. 

So many important and useful algorithms have been developed that learning and 
understanding relationships among them are essential. We finish this section (and this 
book!) with an introduction to the theory of algorithms, with particular focus on intrac¬ 
tability and the P=NP? question that still stands as the key to understanding the practi¬ 
cal problems that we aspire to solve. 
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Event-driven simulation Our first example is a fundamental scientific applica¬ 
tion: simulate the motion of a system of moving particles that behave according to the 
laws of elastic collision. Scientists use such systems to understand and predict proper¬ 
ties of physical systems. This paradigm embraces the motion of molecules in a gas, the 
dynamics of chemical reactions, atomic diffusion, sphere packing, the stability of the 
rings around planets, the phase transitions of certain elements, one-dimensional self- 
gravitating systems, front propagation, and many other situations. Applications range 
from molecular dynamics, where the objects are tiny subatomic particles, to astrophys¬ 
ics, where the objects are huge celestial bodies. 

Addressing this problem requires a bit of high-school physics, a bit of software engi¬ 
neering, and a bit of algorithmics. We leave most of the physics for the exercises at the 
end of this section so that we can concentrate on the topic at hand: using a fundamental 
algorithmic tool (heap-based priority queues) to address an application, enabling cal¬ 
culations that would not otherwise be possible. 

Hard-disc model. We begin with an idealized model of the motion of atoms or mol¬ 
ecules in a container that has the following salient features: 

■ Moving particles interact via elastic collisions with each other and with walls. 

■ Each particle is a disc with known position, velocity, mass, and radius. 

■ No other forces are exerted. 


•k 



advance time to f + dt 




advance time to t + 2dt 

• 0 

roll back time to moment of collision 

Time-driven simulation 


This simple model plays a central role in statistical mechanics, a 
field that relates macroscopic observables (such as temperature and 
pressure) to microscopic dynamics (such as the motion of individ¬ 
ual atoms and molecules). Maxwell and Boltzmann used the model 
to derive the distribution of speeds of interacting molecules as a 
function of temperature; Einstein used the model to explain the 
Brownian motion of pollen grains immersed in water. The assump¬ 
tion that no other forces are exerted implies that particles travel in 
straight lines at constant speed between collisions. We could also 
extend the model to add other forces. For example, if we add fric¬ 
tion and spin, we can more accurately model the motion of familiar 
physical objects such as billiard balls on a pool table. 

Time-driven simulation. Our primary goal is simply to maintain 
the model: that is, we want to be able to keep track of the positions 
and velocities of all the particles as time passes. The basic calcula¬ 
tion that we have to do is the following: given the positions and 
velocities for a specific time t, update them to reflect the situation at 
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dt too small: excessive computation 


a future time t+dt for a specific amount of time dt. Now, if the particles are sufficiently 
far from one another and from the walls that no collision will occur before t+dt, then 
the calculation is easy: since particles travel in a straight-line trajectory, we use each 
particle’s velocity to update its position. The challenge is to take the collisions into 
account. One approach, known as time-driven simulation, is based 
on using a fixed value of dt. To do each update, we need to check 
all pairs of particles, determine whether or not any two occupy the 
same position, and then back up to the moment of the first such 
collision. At that point, we are able to properly update the velocities 
of the two particles to reflect the collision (using calculations that 
we will discuss later). This approach is computationally intensive 
when simulating a large number of particles: if dt is measured in 
seconds (fractions of a second, usually), it takes time proportional 
to N 2 /dt to simulate an N-p article system for 1 second. This cost is 
prohibitive (even worse than usual for quadratic algorithms)—in 
the applications of interest, N is very large and dt is very small. The 
challenge is that if we make dt too small, the computational cost is 
high, and if we make dt too large, we may miss collisions. 



dt too large: may miss collisions 


X 


Fundamental challenge for 
time-driven simulation 


Event-driven simulation. We pursue an alternative approach that 
focuses only on those times at which collisions occur. In particular, we are always in¬ 
terested in the next collision (because the simple update of all of the particle positions 
using their velocities is valid until that time). Therefore, we maintain a priority queue 
of events, where an event is a potential collision sometime in the future, either between 
two particles or between a particle and a wall. The priority associated with each event 
is its time, so when we remove the minimum from the priority queue, we get the next 
potential collision. 


Collision prediction. How do we identify potential collisions? The particle velocities 
provide precisely the information that we need. For example, suppose that we have, at 
time t, a particle of radius s at position (r x , r y ) moving with velocity (v x , v ) in the unit 
box. Consider the vertical wall at x = 1 with y between 0 and 1. Our interest is in the 
horizontal component of the motion, so we can concentrate on the x-component of the 
position r x and the x-component of the velocity v x . If v x is negative, the particle is not 
on a collision course with the wall, but if v x is positive, there is a potential collision with 
the wall. Dividing the horizontal distance to the wall (1 — s — r x ) by the magnitude 
of the horizontal component of the velocity (v x ) we find that the particle will hit the 
wall after dt = (1 — s — r x )/v x time units, when the particle will be at ( 1 — s, r y + v y dt), 
unless it hits some other particle or a horizontal wall before that time. Accordingly, we 


www.it-ebooks.info 


858 


CONTEXT 


resolution (at time t + dt) |—:—\ 

velocity after collision = ( - v x , v y ) 
position after collision = ( 1 - s, r y + Vydt ) 


wall at 
/x=l 


Predicting and resolving a particle-wall collision 

put an entry on the priority queue with priority t + dt (and appropriate information 
describing the particle-wall collision event). The collision-prediction calculations for 
other walls are similar (see exercise 6 . 1 ). The calculation for two particles colliding 
is also similar, but more complicated. Note that it is often the case that the calculation 
leads to a prediction that the collision will not happen (if the particle is moving away 
from the wall, or if two particles are moving away from one another)—we do not need 
to put anything on the priority queue in such cases. To handle another typical situation 
where the predicted collision might be too far in the future to be of interest, we include 
a parameter limit that specifies the time period of interest, so we can also ignore any 
events that are predicted to happen at a time later than 1 i mi t. 

Collision resolution. When a collision does occur, we need to resolve it by applying 
the physical formulas that specify the behavior of a particle after an elastic collision 
with a reflecting boundary or with another particle. In our example where the particle 
hits the vertical wall, if the collision does occur, the velocity of the particle will change 
from (v x , v y ) to (- v x , v ) at that time. The collision-resolution calculations for other 
walls are similar, as are the calculations for two particles colliding, but these are more 
complicated (see exercise 6 . 1 ). 


prediction (at time t) 

dt = time to hit wall 
= distance/velocity 
= (1 -s-r x )lv x 



prediction (at time f) 

particles hit unless one passes 
intersection point before the other 
arrives 




resolution (at time t + dt) 

velocities of both particles 
change after collision 


Predicting and resolving a particle-particle collision 
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Invalidated events. Many of the collisions that we predict do not actu¬ 
ally happen because some other collision intervenes. To handle this situ¬ 
ation, we maintain an instance variable for each particle that counts the 
number of collisions in which it has been involved. When we remove an 
event from the priority queue for processing, we check whether the counts 
corresponding to its particle(s) have changed since the event was creat¬ 
ed. This approach to handling invalidated collisions is the so-called lazy 
approach: when a particle is involved in a colli¬ 
sion, we leave the now-invalid events associated 
with it on the priority queue and essentially ig- 


particle moving 
towards a wall 


\ 


particles moving 
on a collision course 


particle moving 
away from a wall 


Predictable events 


N 


\ 


af 


particles moving 
away from each other 


nore them when they come off. An alternative approach, the 
so-called eager approach, is to remove from the priority queue 
all events involving any colliding particle before calculating all 
of the new potential collisions for that particle. This approach 
requires a more sophisticated priority queue (that implements 
the remove operation). 



one particle reaching 
potential collision point 
before the other 



4 

collision too far 
into the future 


Predictable non-events 


this discussion sets the stage for a full event-driven simula¬ 
tion of particles in motion, interacting according to the physi¬ 
cal laws of elastic collisions. The software architecture is to en¬ 
capsulate the implementation in three classes: a Parti cl e data 
type that encapsulates calculations that involve particles, an 
Event data type for predicted events, and a Col 1 i si onSystem 
client that does the simulation. The cen- 

r . i • i • • ... . two particles on a collision course 

terpiece or the simulation is a Mi nPQ that 
contains events, ordered by time. Next, we 
consider implementations of Particle, 

Event, and Col 1 i si onSystem. 

third particle interferes: no collision 




An invalidated event 
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Particles, exercise 6.1 outlines the implementation of a data type particles, based on 
a direct application of Newton’s laws of motion. A simulation client needs to be able to 
move particles, draw them, and perform a number of calculations related to collisions, 
as detailed in the following API: 

public class Particle 



Parti cle() 

create a new random particle in unit square 


Parti cle( 

create a particle with the given 


double rx, double ry, 

position. 


double vx, double vy, 

velocity, 


double s, 

radius, 


double mass) 

and mass 

voi d 

draw() 

draw the particle 

voi d 

movefdouble dt) 

change position to reflect passage of time dt 

int 

count() 

number of collisions involving this particle 

double 

timeToHit(Particle b) 

time until this particle hits particle b 

double 

timeToHitHorizontalWal1() 

time until this particle hits a horizontal wall 

double 

timeToHitVericaltWal 1 () 

time until this particle hits a vertical wall 

voi d 

bounceOff(Parti cle b) 

change particle velocities to reflect collision 

voi d 

bounceOffHorizontal Wal 1 () 

change velocity to reflect hitting horizontal wall 

voi d 

bounceOffVerticalWal1() 

change velocity to reflect hitting vertical wall 


API for moving-particle objects 


The three timeToHit*() methods all return Doubl e. POSITIVE_INFINITY for the 
(rather common) case when there is no collision course. These methods allow us to 
predict all future collisions that are associated with a given particle, putting an event on 
a priority queue corresponding to each one that happens before a given time 1 i mi t. We 
use the bounce0 method each time that we process an event that corresponds to two 
particles colliding to change the velocities (of both particles) to reflect the collision, and 
the bounce0ff*() methods for events corresponding to collisions between a particle 
and a wall. 
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Events. We encapsulate in a private class the description of the objects to be placed on 
the priority queue (events). The instance variable time holds the time when the event 
is predicted to happen, and the instance variables a and b hold the particles associated 
with the event. We have three different types of events: a particle may hit a vertical wall, 
a horizontal wall, or another particle. To develop a smooth dynamic display of the par¬ 
ticles in motion, we add a fourth event type, a redraw event that is a command to draw 
all the particles at their current positions. A slight twist in the implementation of Event 
is that we use the fact that particle values may be null to encode these four different 
types of events, as follows: 

■ Neither a nor b null: particle-particle collision 

■ a not null and b null: collision between a and a vertical wall 

■ a nul 1 and b not null: collision between b and a horizontal wall 

■ Both a and b nul 1 : redraw event (draw all particles) 

While not the finest object-oriented programming, this convention is a natural one 
that enables straightforward client code and leads to the implementation shown below. 


private class Event implements Comparable<Event> 

{ 

private final double time; 
private final Particle a, b; 
private final int countA, countB; 

public Event(double t, Particle a, Particle b) 

{ // Create a new event to occur at time t involving a and b. 

this.time = t; 
this.a = a; 
this.b = b; 

if (a != null) countA = a.countO; else countA = -1; 
if (b != null) countB = b.count(); else countB = -1; 

} 

public int compareTo(Event that) 

{ 

if (this.time < that.time) return -1; 

else if (this.time > that.time) return +1; 
else return 0; 

} 

public boolean isValid() 

{ 

if (a != null && a.countO != countA) return false; 
if (b != null && b.countO != countB) return false; 
return true; 

} 


Event class for particle simulation 
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A second twist in the implementation of Event is that we maintain the instance vari¬ 
ables countA and countB to record the number of collisions involving each of the par¬ 
ticles at the time the event is created. If these counts are unchanged when the event is 
removed from the priority queue, we can go ahead and simulate the occurrence of the 
event, but if one of the counts changes between the time an event goes on the priority 
queue and the time it leaves, we know that the event has been invalidated and can ig¬ 
nore it. The method i sVal i d () allows client code to test this condition. 

Simulation code. With the computational details encapsulated in Parti cl e and Event, 
the simulation itself requires remarkably little code, as you can see in the implemen¬ 
tation in the class Col I i si onSystem (see page 863 andpage 864). Most of the calculations 
are encapsulated in the predi ctCol 1 i sions() method shown on this page. This meth¬ 
od calculates all potential 
future collisions involv¬ 
ing particle a (either with 
another particle or with 
a wall) and puts an event 
corresponding to each onto 
the priority queue. 

The heart of the simu¬ 
lation is the si mul ate () 
method shown on page 
864. We initialize by calling 
predictCol 1 isions() 
for each particle to fill the 
priority queue with the 
potential collisions involv¬ 
ing all particle-wall and all 
particle-particle pairs. Then we enter the main event-driven simulation loop, which 
works as follows: 

■ Delete the impending event (the one with minimum priority t). 

■ If the event is invalid, ignore it. 

■ Advance all particles to time t on a straight-line trajectory. 

■ Update the velocities of the colliding particle(s). 

■ Use predi ctCol 1 i si ons () to predict future collisions involving the colliding 
particle(s) and insert onto the priority queue an event corresponding to each. 

This simulation can serve as the basis for computing all manner of interesting proper¬ 
ties of the system, as explored in the exercises. For example, one fundamental property 


private void predictCollisions(Particle a, double limit) 
{ 

if (a == null) return; 

for (int i =0; i < parti cles.1ength; i++) 

{ // Put collision with parti cl es[i] on pq. 

double dt = a.timeToHit(particles[i]) ; 
if (t + dt <= limit) 

pq.insert(new Eventft + dt, a, parti cles[i])); 

} 

double dtX = a.timeToHitVerticalWall () ; 
if (t + dtX <= limit) 

pq.insert(new Event(t + dtX, a, null)); 
double dtY = a.timeToHitHorizontalWall () ; 
if (t + dtY <= limit) 

pq.insert(new Event(t + dtY, null, a)); 


Predicting collisions with other particles 
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Event-based simulation of colliding particles (scaffolding) 


public class Col 1isionSystem 

{ 

private class Event implements Comparable<Event> 

{ /* See text. */ } 

private MinPQ<Event> pq; // the priority queue 

private double t = 0.0; // simulation clock time 

private Parti cle[] particles; // the array of particles 

public CollisionSystem(Particle[] particles) 

{ this.parti cles = particles; } 

private void predictCollisions(Particle a, double limit) 

{ /* See text. */ } 

public void redraw(double limit, double Hz) 

{ // Redraw event: redraw all particles. 

StdDraw.cl ear(); 

for(int i = 0; i < particles.length; i++) parti cles[i].draw(); 
StdDraw.show(20); 
if (t < limit) 

pq.insert(new Event(t + 1.0 / Hz, null, null)); 

} 

public void simulate(double limit, double Hz) 

{ /* See next page. */ } 

public static void main(String[] args) 

{ 

StdDraw.show(0); 

int N = Integer.parselntfargs[0]); 

Particlef] particles = new ParticlefN]; 
for (int i =0; i < N; i++) 

parti cles[i] = new Parti cle(); 

Colli sionSystem system = new CollisionSystem(particles); 
system.simulate(10000, 0.5); 

} 

} 


This class is a priority-queue client that simulates the motion of a system of particles over time. 
The main() test client takes a command-line argument N, creates N random particles, creates a 
Col 1 i si onSystem consisting of the particles, and calls si mul ate() to do the simulation. The instance 
variables are a priority queue for the simulation, the time, and the particles. 
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Event-based simulation of colliding particles (primary loop) 


public void simulate(double limit, double Hz) 

{ 

pq = new MinPQ<Event>(); 

for (int i =0; i < parti cles.1ength; i++) 
predictCollisions(particles[i], limit); 
pq.insert(new Event(0, null, null)); // Add redraw event. 


while (! pq.isEmptyO) 

{ // Process one event to drive simulation. 

Event event = pq.delMinO; 

if (!event.isValid()) continue; 

for (int i =0; i < parti cles.1ength; i++) 


parti cles[i].move(event.time 
t = event.time; 


Parti cle 

a 

= event 

■ a, 

b 

= event. 

if 


(a 

!= null 

&& 

b 

!= null) 

el se 

if 

(a 

!= null 

&& 

b 

== null) 

el se 

if 

(a 

== null 

&& 

b 

!= null) 

el se 

if 

(a 

== null 

&& 

b 

== null) 


predictCollisions(a, limit); 
predictCol1isions(b, limit); 

} 

} 


t); // Update particle positions 
// and time. 

; 

a.bounceOff(b); 

a. bounceOffHorizontalWal1(); 

b. bounceOffVertical Wall(); 
redraw(limit, Hz); 


This method represents the main event-driven simulation. First, the priority queue is initialized with 
events representing all predicted future collisions involving each particle. Then the main loop takes 
an event from the queue, updates time and particle positions, and adds new events to reflect changes. 

% java Col 1 i sonSystem 5 a collision 




www.it-ebooks.info 















Event-driven simulation 


865 


of interest is the amount of pressure exerted by the particles against the walls. One way 
to calculate the pressure is to keep track of the number and magnitude of wall colli¬ 
sions (an easy computation based on particle mass and velocity) so that we can easily 
compute the total. Temperature involves a similar calculation. 

Performance. As described at the outset, our interest in event-driven simulation is 
to avoid the computationally intensive inner loop intrinsic in time-driven simulation. 

Proposition A. An event-based simulation of N colliding particles requires at most 
N 2 priority queue operations for initialization, and at most N priority queue op¬ 
erations per collision (with one extra priority queue operation for each invalid 
collision). 

Proof Immediate from the code. 


Using our standard guaranteed-logarithmic-time-per operation priority-queue imple¬ 
mentation from section 2.4, the time needed per collision is linearithmic. Simulations 
with large numbers of particles are therefore quite feasible. 

event-driven simulation applies to countless other domains that involve physical 
modeling of moving objects, from molecular modeling to astrophysics to robotics. 
Such applications may involve extending the model to add other kinds of bodies, to 
operate in three dimensions, to include other forces, and in many other ways. Each ex¬ 
tension involves its own computational challenges. This event-driven approach results 
in a more robust, accurate, and efficient simulation than many other alternatives that 
we might consider, and the efficiency of the heap-based priority queue enables calcula¬ 
tions that might not otherwise be possible. 

Simulation plays a vital role in helping researchers to understand properties of the 
natural world in all fields of science and engineering. Applications ranging from manu¬ 
facturing processes to biological systems to financial systems to complex engineered 
structures are too numerous to even list here. For a great many of these applications, 
the extra efficiency afforded by the heap-based priority queue data type or an efficient 
sorting algorithm can make a substantial difference in the quality and extent that are 
possible in the simulation. 
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B-trees In chapter 3, we saw that algorithms that are appropriate for accessing 
items from huge collections of data are of immense practical importance. Searching is 
a fundamental operation on huge data sets, and such searching consumes a significant 
fraction of the resources used in many computing environments. With the advent of 
the web, we have the ability to access a vast amount of information that might be rele¬ 
vant to a task—our challenge is to be able to search through it efficiently. In this section, 
we describe a further extension of the balanced-tree algorithms from section 3.3 that 
can support external search in symbol tables that are kept on a disk or on the web and 
are thus potentially far larger than those we have been considering (which have to fit in 
addressable memory). Modern software systems are blurring the distinction between 
local files and web pages, which may be stored on a remote computer, so the amount 
of data that we might wish to search is virtually unlimited. Remarkably, the methods 
that we shall study can support search and insert operations on symbol tables contain¬ 
ing trillions of items or more using only four or five references to small blocks of data. 

Cost model. Data storage mechanisms vary widely and continue to evolve, so we use 
a simple model to capture the essentials. We use the term page to refer to a contiguous 
block of data and the term probe to refer to the first access to a page. We assume that 
accessing a page involves reading its contents into local memory, so that subsequent ac¬ 
cesses are relatively inexpensive. A page could be a file on your local computer or a web 
page on a distant computer or part of a file on 
a server, or whatever. Our goal is to develop 
search implementations that use a small num¬ 
ber of probes to find any given key. We avoid 
making specific assumptions about the page 
size and about the ratio of the time required 
for a probe (which presumably requires com¬ 
municating with a distant device) to the time 
required, subsequently, to access items within 
the block (which presumably happens in a local processor). In typical situations, these 
values are likely to be on the order of 100 or 1,000 or 10,000; we do not need to be more 
precise because the algorithms are not highly sensitive to differences in the values in the 
ranges of interest. 

B-trees. The approach is to extend the 2-3 tree data structure described in section 
3.3, with a crucial difference: rather than store the data in the tree, we build a tree with 
copies of the keys, each key copy associated with a link. This approach enables us to 
more easily separate the index from the table itself, much like the index in a book. As 
with 2-3 trees, we enforce upper and lower bounds on the number of key-link pairs 


B-tree cost model. When study¬ 
ing algorithms for external 
searching, we count page accesses 
(the number of times a page is ac¬ 
cessed, for read or write). 
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that can be in each node: we choose a parameter M (an even number, by convention) 
and build multiway trees where every node must have at most M — 1 key-link pairs (we 
assume that M is sufficiently small that an M -way node will fit on a page) and at least 
M /2 key-link pairs (to provide the branching that we need to keep search paths short), 
except possibly the root, which can have fewer than Ml 2 key-link pairs but must have 
at least 2 . Such trees were named B-trees by Bayer and McCreight, who, in 1970 , were 
the first researchers to consider the use of multiway balanced trees for external search¬ 
ing. Some people reserve the term B-tree to describe the exact data structure built by 
the algorithm suggested by Bayer and McCreight; we use it as a generic term for data 
structures based on multiway balanced search trees with a fixed page size. We specify 
the value of M by using the terminology “ B-tree of order M.” In a B-tree of order 4 , each 
node has at most 3 and at least 2 key-link pairs; in a B-tree of order 6, each node has at 
most 5 and at least 3 link pairs (except possibly the root, which could have 2 key-link 
pairs), and so forth. The reason for the exception at the root for larger M will become 
clear when we consider the construction algorithm in detail. 

Conventions. To illustrate the basic mechanisms, we consider an (ordered) SET imple¬ 
mentation (with keys and no values). Extending to provide an ordered ST to associate 
keys with values is an instructive exercise (see exercise 6.16). Our goal is to support 
add() and contai ns() for a set of keys that could be huge. We use ordered keys be¬ 
cause we are generalizing search trees, which are based on ordered keys. Extending our 
implementation to support other ordered operations is also an instructive exercise. In 
external searching applications, it is common to keep the index separate from the data. 
For B-trees, we do so by using two different kinds of nodes: 

■ Internal nodes, which associate copies of keys with pages 

■ External nodes, which have references to the actual data 

Every key in an internal node is associated with another node that is the root of a tree 
containing all keys greater than or equal to that key and less than the next largest key, if 



client keys (black) 0 // nodes except the root are 3-, 4- or 5-nodes 

are in external nodes 

Anatomy of a B-tree set (M = 6) 
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any. It is convenient to use a special key, known as a sentinel, that is defined to be less 
than all other keys, and to start with a root node containing that key, associated with 
the tree containing all the keys. The symbol table does not contain duplicate keys, but 
we use copies of keys (in internal nodes) to guide the search. (In our examples, we use 
single-letter keys and the character * as the sentinel that is less than all other keys.) 
These conventions simplify the code somewhat and thus represent a convenient (and 
widely used) alternative to mixing all the data with links in the internal nodes, as we 
have done for other search trees. 

Search and insert. Search in a B-tree is based on recursively searching in the unique 
subtree that could contain the search key. Every search ends in an external node that 
contains the key if and only if it is in the set. We might also terminate a search hit when 
encountering a copy of the search key in an internal node, but we always search to an 
external node because doing so simplifies extending the code to an ordered symbol- 
table implementation (also, this event rarely happens when M is large). To be specific, 
consider searching in a B-tree of order 6: it consists of 3-nodes with 3 key-link pairs, 
4-nodes with 4 key-link pairs, and 5-nodes with 5 key-link pairs, with possibly a 2-node 
at the root. To search, we start at the root and move from one node to the next by find¬ 
ing the proper interval for the search key in the current node and then exiting through 
the corresponding link to get to the next node. Eventually, the search process leads us to 
a page containing keys at the bottom of the tree. We terminate the search with a search 
hit if the search key is in that page; we terminate with a search miss if it is not. As with 

2- 3 trees, we can use recursive code to insert a new key at the bottom of the tree. If there 
is no room for the key, we allow the node at the bottom to temporarily overflow (be¬ 
come a 6-node) and then split 6-nodes on the way up the tree, after the recursive call. If 
the root is an 6-node, we split it into a 2-node connected to two 3-nodes; elsewhere in 
the tree, we replace any k-node attached to a 6-node by a (k+l)-node attached to two 

3- nodes. Replacing 3 by M/2 and 6 by M in this description converts it into a descrip¬ 
tion of search and insert for B-trees of order M and leads to the following definition: 

Definition. A B-tree of order M (where M is an even positive integer) is a tree that 
either is an external k-node (with k keys and associated information) or comprises 
internal fc-nodes (each with k keys and k links to B-trees representing each of the k 
intervals delimited by the keys), having the following structural properties: every 
path from the root to an external node must be the same length (perfect balance); 
and k must be between 2 and M — 1 at the root and between Ml 2 and M— 1 at 
every other node. 
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search for E in 
this external node 


Searching in a B-tree set (M = 6) 


inserting A 
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CONTEXT 


Representation. As just discussed, we have a great deal of freedom in choosing con¬ 
crete representations for nodes in B-trees. We encapsulate these choices in a Page API 
that associates keys with links to Page objects and supports the operations that we 
need to test for overfull pages, split them, and distinguish between internal and exter¬ 
nal pages. You can think of a Page as a symbol table, kept externally (in a file on your 
computer or on the web). The terms open and close in the API refer to the process of 
bringing an external page into internal memory and writing its contents back out (if 
necessary). The put() method for internal pages is a symbol-table operation that asso¬ 
ciates the given page with the minimum key in the tree rooted at that page. The put 0 
and containsO methods for external pages are like their corresponding SET opera¬ 
tions. The workhorse of any implementation is the spl i t() method, which splits a full 
page by moving the Ml 2 key-value pairs of rank greater than M/2 to a new Page and 
returns a reference to that page, exercise 6.15 discusses an implementation of Page 
using Bi narySearchST, which implements B-trees in memory, like our other search 
implementations. On some systems, this might suffice as an external searching imple¬ 
mentation because a virtual-memory system might take care of disk references. More 
typical practical implementations might involve hardware-specific code that reads and 


public class Page<Key> 



Page(boolean bottom) 

create and open a page 

voi d 

close() 

close a page 

voi d 

add(Key key) 

put key into the (external) page 

voi d 

add(Page p) 

open p and put an entry into this 
(internal) page that associates the 
smallest key in p with p 

boolean 

isExternal() 

is this page external? 

boolean 

contains(Key key) 

is key in the page? 

Page 

next(Key key) 

the subtree that could contain the key 

boolean 

isFull() 

has the page overflowed? 

Page 

splitO 

move the highest-ranking half of the 
keys in the page to a new page 

Iterable<Key> 

keys() 

iterator for the keys on the page 


API for a B-tree page 
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writes pages, exercise 6.19 encourages you to think about implementing Page using 
web pages. We ignore such details here in the text to emphasize the utility of the B-tree 
concept in a broad variety of settings. 

With these preparations, the code for BT reeSET on page 872 is remarkably simple. For 
containsO, we use a recursive method that takes a Page as argument and handles 
three cases: 

■ If the page is external and the key is in the page, return true. 

■ If the page is external and the key is not in the page, return fal se. 

■ Otherwise, do a recursive call for the subtree that could contain the key. 

For put() we use the same recursive structure, but insert the key at the bottom if it is 
not found during the search and then split any full nodes on the way up the tree. 

Performance. The most important property of B-trees is that for reasonable values of 
the parameter M the search cost is constant, for all practical purposes: 

Proposition B. A search or an insertion in a B-tree of order M with N items requires 
between log M N and log M/2 N probes—a constant number, for practical purposes. 

Proof This property follows from the observation that all the nodes in the interior 
of the tree (nodes that are not the root and are not external) have between M/2 
and M — 1 links, since they are formed from a split of a full node with M keys and 
can only grow in size (when a child is split). In the best case, these nodes form a 
complete tree of branching factor M — 1, which leads immediately to the stated 
bound. In the worst case, we have a root with two entries each of which refers to a 
complete tree of degree M/2. Taking the logarithm to the base M results in a very 
small number—for example, when M is 1,000, the height of the tree is less than 4 
for N less than 62.5 billion. 


In typical situations, we can reduce the cost by one probe by keeping the root in in¬ 
ternal memory. For searching on disk or on the web, we might take this step explicitly 
before embarking on any application involving a huge number of searches; in a virtual 
memory with caching, the root node will be the one most likely to be in fast memory, 
because it is the most frequently accessed node. 

Space. The space usage of B-trees is also of interest in practical applications. By con¬ 
struction, the pages are at least half full, so, in the worst case, B-trees use about double 
the space that is absolutely necessary for keys, plus extra space for links. For random 
keys, A. Yao proved in 1979 (using mathematical analysis that is beyond the scope of 
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ALGORITHM 6.12 B-tree set implementation 


public class BTreeSET<Key extends Comparable<Key» 

{ 

private Page root = new Page(true); 

public BTreeSET(Key sentinel) 

{ putCsentinel) ; } 

public boolean contains(Key key) 

{ return contains(root, key); } 

private boolean contains(Page h, Key key) 

{ 

if (h.isExternal()) return h.contains(key); 
return contains(h.next(key), key); 

} 

public void add(Key key) 

{ 

put (root, key); 
if (root.isFull()) 

{ 

Page lefthalf = root; 

Page righthalf = root.split(); 
root = new Page(false); 
root.put(lefthalf); 
root.put(righthalf); 

} 

} 

public void add(Page h, Key key) 

{ 

if (h.isExternal()) { h.put(key); return; } 

Page next = h.next(key); 
put(next, key); 
if (next.isFull()) 

h.put(next.split()); 
next.close(); 

} 

} 


This B-tree implementation implements multiway balanced search trees as described in the text, us¬ 
ing a Page data type that supports search by associating keys with subtrees that could contain the key 
and supports insertion by including a test for overflow and a page split method. 


www.it-ebooks.info 






B-trees 


873 



Building a large B-tree 
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CONTEXT 


this book) that the average number of keys in a node is about M In 2, so about 44 per¬ 
cent of the space is unused. As with many other search algorithms, this random model 
reasonably predicts results for key distributions that we observe in practice. 

the implications of proposition b are profound and worth contemplating. Would 
you have guessed that you can develop a search implementation that can guarantee a 
cost of four or five probes for search and insert in files as large as you can reason¬ 
ably contemplate needing to process? B-trees are widely used because they allow us to 
achieve this ideal. In practice, the primary challenge to developing an implementation 
is ensuring that space is available for the B-tree nodes, but even that challenge becomes 
easier to address as available storage space increases on typical devices. 

Many variations on the basic B-tree abstraction suggest themselves immediately. 
One class of variations saves time by packing as many page references as possible in 
internal nodes, thereby increasing the branching factor and flattening the tree. Another 
class of variations improves storage efficiency by combining nodes with siblings before 
splitting. The precise choice of variant and algorithm parameter can be engineered to 
suit particular devices and applications. Although we are limited to getting a small con¬ 
stant factor improvement, such an improvement can be of significant importance for 
applications where the table is huge and/or huge numbers of transactions are involved, 
precisely the applications for which B-trees are so effective. 
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Suffix arrays Efficient algorithms for string processing play a critical role in com¬ 
mercial applications and in scientific computing. From the countless strings that define 
web pages that are searched by billions of users to the extensive genomic databases that 
scientists are studying to unlock the secret of life, computing applications of the 21st 
century are increasingly string-based. As usual, some classic algorithms are effective, 
but remarkable new algorithms are being developed. Next, we describe a data structure 
and an API that support some of these algorithms. We begin by describing a typical 
(and a classic) string-processing problem. 

Longest repeated substring. What is the longest substring that appears at least 
twice in a given string? For example, the longest repeated substring in the string 
"to be or not to be" is the string "to be". Think briefly about how you might solve 
it. Could you find the longest repeated substring in a string that has millions of char¬ 
acters? This problem is simple to state and has many important applications, including 
data compression, cryptography, and computer-assisted music analysis. For example, 
a standard technique used in the development of large software systems is refactoring 
code. Programmers often put together new programs by cutting and pasting code from 
old programs. In a large program built over a long period of time, replacing duplicate 
code by function calls to a single copy of the code can make the program much easier 
to understand and maintain. This improvement can be accomplished by finding long 
repeated substrings in the program. Another application is found in computational 
biology. Are substantial identical fragments to be found within a given genome? Again, 
the basic computational problem underlying this question is to find the longest repeat¬ 
ed substring in a string. Scientists are typically interested in more detailed questions 
(indeed, the nature of the repeated substrings is precisely what scientists seek to under¬ 
stand), but such questions are certainly no easier to answer than the basic question of 
finding the longest repeated substring. 

Brute-force solution. As a warmup, consider the following simple task: given two 
strings, find their longest common prefix (the longest substring that is a prefix of both 
strings). For example, the longest 
common prefix of acctgttaac and 
accgttaa is acc. The code at right 
is a useful starting point for address¬ 
ing more complicated tasks: it takes 
time proportional to the length of 
the match. Now, how do we find the 
longest repeated substring in a given 
string? With lcpQ, the following 


private static int lcp(String s, String t) 

{ 

int N = Math.min(s.length?), t.length?)); 
for ?int i =0; i < N; i++) 

if ?s.charAt?i) != t.charAtfi)) return i; 
return N; 

} 

Longest common prefix of two strings 
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brute-force solution immediately suggests itself: we compare the substring starting at 
each string position i with the substring starting at each other starting position j, keep¬ 
ing track of the longest match found. This code is not useful for long strings, because 
its running time is at least quadratic in the length of the string: as usual, the number of 
different pairs i and j is N (N— 1)/2, so the number of calls on 1 cp() for this approach 
would be ~N 2 I2. Using this solution for a genomic sequence with millions of characters 
would require trillions of 1 cp () calls, which is infeasible. 


input string 

012345678 91011121314 

aacaagtttacaagc 


suffixes 


Suffix sort solution. The following clever approach, which takes advantage of sorting 
in an unexpected way, is an effective way to find the longest repeated substring, even in 
a huge string: we use Java’s substri ng() meth¬ 
od to make an array of strings that consists of 
the suffixes of s (the substrings starting at each 
position and going to the end), and then we sort 
this array. The key to the algorithm is that ev¬ 
ery substring appears somewhere as a prefix of 
one of the suffixes in the array. After sorting, the 
longest repeated substrings will appear in adja¬ 
cent positions in the array. Thus, we can make 
a single pass through the sorted array, keeping 
track of the longest matching prefixes between 
adjacent strings. This approach is significantly 
more efficient than the brute-force method, but 
before implementing and analyzing it, we con¬ 
sider another application of suffix sorting. 


10 

n 

12 

13 

14 


aacaagtttacaagc 

acaagtttacaagc 

caagtttacaagc 

aagtttacaagc 

agtttacaagc 

gtttacaagc 

tttacaagc 

ttacaagc 

t a c a a g c 

a c a a g c 

c a a g c 

a a g c 

age 

9 c 

c 


sorted suffixes 

o aacaagtttacaagc 
n a a g c 

3 aagtttacaagc 

9 a c a a g c 

1 acaagtttacaagc 

12 a g c 

4 agtttacaagc 

14 c 

10 c a a g c 

2 caagtttacaagc 

13 g c 

5 gtttacaagc 
8 tacaagc 

7 ttacaagc 

6 tttacaagc 

longest repeated substring 


aacaagtttacaagc 


Computing the LRS by sorting suffixes 
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Indexing a string. When you are trying to find a particular substring within a large 
text—for example, while working in a text editor or within a page you are viewing with 
a browser—you are doing a substring search, the problem we considered in section 
5 . 3 . For that problem, we assume the text to be relatively large and focus on preprocess¬ 
ing the substring, with the goal of being able to efficiently find that substring in any 
given text. When you type search keys into your web browser, you are doing a search 
with string keys, the subject of section 5 . 2 . Your search engine must precompute an 
index, since it cannot afford to scan all the pages in the web for your keys. As we dis¬ 
cussed in section 3.5 (see Fi 1 elndex on page 501), this would ideally be an inverted in¬ 
dex associating each possible search 
string with all web pages that con¬ 
tain it—a symbol table where each 
entry is a string key and each value 
is a set of pointers (each pointer giv¬ 
ing the information necessary to lo¬ 
cate an occurrence of the key on the 
web—perhaps a URL that names 
a web page and an integer offset 
within that page). In practice, such 
a symbol table would be far too big, 
so your search engine uses various 
sophisticated algorithms to reduce 
its size. One approach is to rank web 
pages by importance (perhaps using 
an algorithm like the PageRank al¬ 
gorithm that we discussed on page 
507) andworkonlywithhighly-ranked 
pages, not all pages. Another ap¬ 
proach to cutting down on the size 
of a symbol table to support search with string keys is to associate URLs with words 
(substrings delimited by whitespace) as keys in the precomputed index. Then, when 
you search for a word, the search engine can use the index to find the (important) 
pages containing your search keys (words) and then use substring search within each 
page to find them. But with this approach, if the text were to contain "everythi ng" 
and you were to search for "thi ng", you would not find it. For some applications, it 
is worthwhile to build an index to help find any substring within a given text. Doing 
so might be justified for a linguistic study of an important piece of literature, for a ge¬ 
nomic sequence that might be an object of study for many scientists, or just for a widely 


symbol-table search with string keys: 
find the pages containing the key 



Idealized view of a typical web search 
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accessed web page. Again, ideally, the index 
would associate all possible substrings of the 
text string with each position where it occurs 
in the text string, as depicted at right. The ba¬ 
sic problem with this ideal is that the number 
of possible substrings is too large to have a 
symbol-table entry for each of them (an N- 
character text has N(N~ l)/2 substrings). 

The table for the example at right would need 
entries for b, be, bes, best, best o, best of, 
e, es, est, est o, est of, s, st, st o, st of, 
t, t o, t of, o, of, and many, many other substrings. Again, we can use a suffix sort to 
address this problem in a manner analogous to our first symbol-table implementation 
using binary search, in section 3 . 1 . We consider each of the N suffixes to be keys, create 
a sorted array of our keys (the suffixes), and use binary search to search in that array, 
comparing the search key with each suffix. 


key 


value 




best of times 



1 it was 





r it was the 
- best of 
, it was thi 

worst of times 
r it was th< 
of wisdom 
„it was he age 
of foolishness 
► it was th< 
epoch of 
belief 


Idealized view of a text-string index 


suffixes 

it was the best of times it was the 
t was the best of times it was the 
was the best of times it was the 
was the best of times it was the 
4 as the best of times it was the 
s the best of times it was the 
the best of times it was the 
the best of times it was the 
he best of times it was the 
e best of times it was the 


sorted suffix array 

i index(i) lcp(i) 

0 10 0 
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the best of times it was the 
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was the best of times it was the 
as the 

as the best of times it was the- 


10 

best of times it was the 


10^J 

f 11 

0 

best of times it was the 

11 

best of times it was the 

i ndex(9) ''' 

<fi 

34 

0 

e 

12 

est of times it was the 

12 

9 

1 

e best of times it was the 

13 

st of times it was the 


13 

22 

1 

es it was the 

14 

t of times it was the 


14 

12 

2 

est of times it was the 

15 

of times it was the 


15 

17 

0 

f times it was the 

16 

of times it was the 


16 

33 

0 

he 

17 

f times it was the 


17 

8 

2 

he best of times it was the 

18 

times it was the 


18 

20 

0 

imes it was the 

19 

times it was the 


19 

25 

1 

it was the 

20 

imes it was the 


20 

0 

10 

it was the best of times it was the 

21 

mes it was the 


21 


r 0 

mes it was the 

22 

es it was the 


22 ^ 

■^16 

0 

of times it was the 

23 

s it was the 

lcp(20)^ 
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s it was the 

24 

it was the 

24 

30 

2 
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25 

it was the 


25 

5 
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s the best of times it was the 

26 

t was the 
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13 
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27 

was the 
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14 
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t of times it was the 

28 

was the 


28 

26 
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t was the 

29 

as the 


29 

1 

9 

t was the best of times it was the 

30 

s the 

rank("th") —>- 

30 

32 
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31 
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31 

7 

3 

the best of times it was the 

32 

the 


32 

19 

1 

times it was the 

33 
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33 

28 

0 
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34 
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34 

3 

7 

was the best of times it was the 


.select(9) 


Binary search in a suffix array 
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intervals containing 
"th" found by rank() 
during binary search 
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API and client code. To support client code to solve these two problems, we artic¬ 
ulate the API shown below. It includes a constructor; a length () method; methods 
sel ectO and i ndex(), which give the string and index of the suffix of a given rank in 
the sorted list of suffixes; a method 1 cp() that gives the length of the longest common 
prefix of each suffix and the one preceding it in the sorted list; and a method rank() 
that gives the number of suffixes less than the given key (just as we have been using 
since we first examined binary search in chapter i). We use the term suffix array to 
describe the abstraction of a sorted list of suffix strings, without necessarily committing 
to use an array of strings as the underlying data structure. 


public class SuffixArray 


int 
String 
int 
int 
int 


SuffixArray (Stri ng text) build suffix array for text 


lengthO 
select(int i) 
index(int i) 
lcp(int i) 
rank(String key) 


length of text 

i th in the suffix array 
(i between 0 and N-l) 

index of select(i) (i between 0 and N-l) 

length of longest common prefix of sel ect(i) 
and sel ect(i -1) (i between 1 and N-l) 

number of suffixes less than key 

Suffix array API 


In the example on the facing page, select (9) is "as the best of times. . .",index(9) 
is 4, lcp(20) is 10 because "it was the best of times ..." and "it was the" have 
the common prefix "i t was the" which is of length 10, and rank("th") is 30. Note 
also that the sel ect( rank (key)) is the first possible suffix in the sorted suffix list that 
has key as prefix and that all other occurrences of key in the text immediately follow 
(see the figure on the opposite page). With this API, the client code on the next two 
pages is immediate. LRS (page880) finds the longest repeated substring in the text on stan¬ 
dard input by building a suffix array and then scanning through the sorted suffixes to 
find the maximum 1 cp() value. KWIC (page 881) builds a suffix array for the text named 
as command-line argument, takes queries from standard input, and prints all occur¬ 
rences of each query in the text (including a specified number of characters before and 
after to give context). The name KWIC stands for keyword-in-context search, a term 
dating at least to the 1960s. The simplicity and efficiency of this client code for these 
typical string-processing applications is remarkable, and testimony to the importance 
of careful API design (and the power of a simple but ingenious idea). 
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public class LRS 

{ 

public static void mai n(String[] args) 

{ 

String text = Stdln. readAl 1 () ; 
int N = text.1ength(); 

SuffixArray sa = new SuffixArray(text); 

String 1 rs = 

for (int i =1; i < N; i++) 

{ 

int length = sa.lcp(i); 
if (length > substring. lengthO) 

1 rs = sa.select(i) .substring(0, length); 

} 

StdOut. pri ntl n(l rs) ; 

} 

} 

Longest repeated substring client 


% more tinyTale.txt 

it was the best of times it was the worst of times 
it was the age of wisdom it was the age of foolishness 
it was the epoch of belief it was the epoch of incredulity 
it was the season of light it was the season of darkness 
it was the spring of hope it was the winter of despair 

% java LRS < tinyTale.txt 
st of times it was the 
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public class KWIC 

{ 

public static void main(String[] args) 

{ 

In in = new ln(args[0]); 

int context = Integer.parselnt(args[1]); 

String text = in.readAll().replaceAll("\\s+", " 
int N = text.lengthO ; 

SuffixArray sa = new SuffixArray(text); 

while (Stdln.hasNextLine()) 

{ 

String q = Stdln. readLi ne() ; 

for (int i = sa.rank(q); i < N && sa.select(i) . startsWi th(q); i++) 

{ 

int from = Math.max(0, sa.index(i) - context); 

int to = Math .mi n(N-l, from + q.lengthO + 2*context) ; 

StdOut.println(text.substring(from, to)); 

} 

StdOut. pri ntln() ; 

} 

} 

} 

Keyword-in-context indexing client 


% java KWIC tale.txt 15 
search 

o st giless to 
her unavailing 
le and gone in 
t provinces in 
dispersing in 
n that bed and 


search for contraband 
search for your fathe 
search of her husband 
search of impoverishe 
search of other carri 
search the straw hold 


better thing 

t is a far far better thing that i do than 
some sense of better things else forgotte 
was capable of better things mr carton ent 
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Implementation. The code on the facing page is a straightforward implementation of 
the Suffi xArray API. Its instance variables are an array of strings and (for economy 
in code) a variable N that holds the length of the array (the length of the string and its 
number of suffixes). The constructor builds the suffix array and sorts it, so sel ect (i) 
just returns suffi xes [i ]. The implementation of i ndex() is also a one-liner, but it is a 
bit tricky, based on the observation that the length of the suffix string uniquely determines 
its starting point. The suffix of length N starts at position 0, the suffix of length N-l starts 
at position 1, the suufix of length N-2 starts at position 2, and so forth, so i ndex(i) 
just returns N - suffi xes [i] .length(). The implementation of lcp() is immedi¬ 
ate,given the static method 1 cp() on page 875, and rank() is virtually the same as our 
implementation of binary search for symbol tables, on page 381. Again, the simplicity and 
elegance of this implementation should not mask the fact that it is a sophisticated algo¬ 
rithm that enables solution of important problems like the longest repeated substring 
problem that would otherwise seem to be infeasible. 

Performance. The efficiency of suffix sorting depends on the fact that Java substring 
extraction uses a constant amount of space—each substring is composed of standard 
object overhead, a pointer into the original, and a length. Thus, the size of the index 
is linear in the size of the string. This point is a bit counterintuitive because the total 
number of characters in the suffixes is ~N 2 1 2, a quadratic function of the size of the 
string. Moreover, that quadratic factor gives one pause when considering the cost of 
sorting the suffix array. It is very important to bear in mind that this approach is effec¬ 
tive for long strings because of the Java representation for strings: when we exchange 
two strings, we are exchanging only references, not the whole string. Now, the cost of 
comparing two strings may be proportional to the length of the strings in the case when 
their common prefix is very long, but most comparisons in typical applications involve 
only a few characters. If so, the running time of the suffix sort is linearithmic. For ex¬ 
ample, in many applications, it is reasonable to use a random string model: 

Proposition C. Using 3-way string quicksort, we can build a suffix array from a 
random string of length N with space proportional to N and ~ 2/Vln N character 
compares, on the average. 

Discussion: The space bound is immediate, but the time bound is follows from 
a a detailed and difficult research result by P. Jacquet and W. Szpankowski, which 
implies that the cost of sorting the suffixes is asymptotically the same as the cost of 
sorting N random strings (see proposition e on page 723). 
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ALGORITHM 6.13 Suffix array (elementary implementation) 


public class SuffixArray 

{ 

private final String[] suffixes; 
private final int N; 


// suffix array 

// length of string (and array) 


public SuffixArrayCString s) 

{ 

N = s.lengthO ; 

suffixes = new String[N]; 

for (int i =0; i < N; i++) 

suffixes[i] = s.substring(i); 

Quick3way.sort(suffixes); 

} 

public int lengthO { return N; } 

public String select(int i) { return suffixesfi]; } 

public int indexfint i) { return N - suffixes[i] .lengthO ; } 

private static int IcpCString s, String t) 

// See page page 875. 

public int lcp(int i) 

{ return 1cpfsuffixes[i], suffixes[i-l]); } 

public int rank(String key) 

{ // binary search 

int lo = 0, hi = N - 1; 
while (lo <= hi) 

{ 

int mid = lo + (hi - lo) / 2; 

int cmp = key.compareTo(suffixes[mid]); 

if (cmp < 0) hi = mid - 1; 

else if (cmp >0) lo = mid + 1; 

else return mid; 

} 

return lo; 

} 


This implementation of our Suffi xArray API depends for its efficiency on the fact that Java Stri ng 
values are immutable, so that substrings are constant-size references and substring extraction takes 
constant time (see text). 
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Improved implementations. Our elementary implementation of SuffixArray has 
poor worst-case performance. For example, if all the characters are equal, the sort ex¬ 
amines every character in each substring and 
thus takes quadratic time. For strings of the 
type we have been using as examples, such as 
genomic sequences or natural-language text, 
this is not likely to be problematic, but the al¬ 
gorithm can be slow for texts with long runs of 
identical characters. Another way of looking at 
the problem is to observe that the cost of find¬ 
ing the longest repeated substring is quadratic 
in the length of the substring because all of the 
prefixes of the repeat need to be checked (see 
the diagram at right). This is not a problem for 
a text such as A Tale of Two Cities, where the 
longest repeat 

"s dropped because it would have 
been a bad thing for me in a 
worldly point of view i" 

has just 84 characters, but it is a serious prob- 

comparison cost is at least 

lem for genomic data, where long repeated sub- 1 + 2 + ... + m ~ m 2 /i 
strings are not unusual. How can this quadratic LRS cost is quadratic in repeat length 
behavior for repeat searching be avoided? Re¬ 
markably, research by R Weiner in 1973 showed 

that it is possible to solve the longest repeated substring problem in guaranteed linear time. 
Weiner’s algorithm was based on building a suffix tree data structure (essentially a 
trie for suffixes). With multiple pointers per character, suffix trees consume too much 
space for many practical problems, which led to the development of suffix arrays. In the 
1990s, U. Manber and E. Myers presented a linearithmic algorithm for building suffix 
arrays directly and a method that does preprocessing at the same time as the suffix sort 
to support constant-time 1 cp O. Several linear-time suffix sorting algorithms have been 
developed since. With a bit more work, the Manber-Myers implementation can also 
support a two-argument 1 cp() that finds the longest common prefix of two given suf¬ 
fixes that are not necessarily adjacent in guaranteed constant time, again a remarkable 
improvement over the straightforard implementation. These results are quite surpris¬ 
ing, as they achieve efficiencies quite beyond what you might have expected. 


input string 

aacaagtttacaagc 

suffixes of longest repeat (M = 5) 
a c a a g 


c a a g 
a a g 
a g 
9 


all appear at least 
twice as a prefix of 
a suffix string 


sorted suffixes of input 

aacaagtttacaagc 
a a g c 

aagtttacaagc 
a c a a g c 

acaagtttacaagc 
age 

agtttacaagc 
c 

c a a g c 

caagtttacaagc 
9 c 

gtttacaagc 
t a c a a g c 
ttacaagc 
tttacaagc 
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Proposition D. With suffix arrays, we can solve both the suffix sorting and longest 
repeated substring problems in linear time. 

Proof: The remarkable algorithms for these tasks are just beyond our scope, but 
you can find on the booksite code that implements the Suff i xAr ray constructor 
in linear time and 1 cp() queries in constant time. 


A SuffixArray implementation based on these ideas supports efficient solutions of 
numerous string-processing problems, with simple client code, as in our LRS and KWIC 
examples. 

suffix arrays are the culmination of decades of research that began with the de¬ 
velopment of tries for KWIC indices in the 1960s. The algorithms that we have dis¬ 
cussed were worked out by many researchers over several decades in the context of solv¬ 
ing practical problems ranging from putting the Oxford English Dictionary online to the 
development of the first web search engines to sequencing the human genome. This 
story certainly helps put the importance of algorithm design and analysis in context. 
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Network-flow algorithms Next, we consider a 
graph model that has been successful not just because it 
provides us with a simply stated problem-solving model 
that is useful in many practical applications but also be¬ 
cause we have efficient algorithms for solving problems 
within the model. The solution that we consider illustrates 
the tension between our quest for implementations of gen¬ 
eral applicability and our quest for efficient solutions to 
specific problems. The study of network-flow algorithms is 
fascinating because it brings us tantalizingly close to com¬ 
pact and elegant implementations that achieve both goals. 
As you will see, we have straightforward implementations 
that are guaranteed to run in time proportional to a poly¬ 
nomial in the size of the network. 

The classical solutions to network-flow problems are 
closely related to other graph algorithms that we studied 
in chapter 4 , and we can write surprisingly concise pro¬ 
grams that solve them, using the algorithmic tools we have 
developed. As we have seen in many other situations, good 
algorithms and data structures can lead to substantial re¬ 
ductions in running times. Development of better imple¬ 
mentations and better algorithms is still an area of active 
research, and new approaches continue to be discovered. 

A physical model. We begin with an idealized physical 
model in which several of the basic concepts are intui¬ 
tive. Specifically, imagine a collection of interconnected oil 
pipes of varying sizes, with switches controlling the direc¬ 
tion of flow at junctions, as in the example illustrated at 
right. Suppose further that the network has a single source 
(say, an oil field) and a single sink (say, a large refinery) to 
which all the pipes ultimately connect. At each vertex, the 
flowing oil reaches an equilibrium where the amount of oil 
flowing in is equal to the amount flowing out. We measure 
both flow and pipe capacity in the same units (say, gallons 
per second). If every switch has the property that the total 
capacity of the ingoing pipes is equal to the total capacity 
of the outgoing pipes, then there is no problem to solve: we 
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inflow equals outflow 
at every vertex ** 

(except the source 
and the sink) 


Local equilibrium in a flow network 


simply fill all pipes to full capacity. Otherwise, not all pipes are 
full, but oil flows through the network, controlled by switch 
settings at the junctions, satisfying a local equilibrium con¬ 
dition at the junctions: the amount of oil flowing into each 
junction is equal to the amount of oil flowing out. For ex¬ 
ample, consider the network in the diagram on the opposite 
page. Operators might start the flow by opening the switches 
along the path 0->l->3->5, which can handle 2 units of flow, 
then open switches along the path 0->2->4->5 to get another 

unit of flow in the network. Since 0->l, 2->4, and 3->5 are full, there is no direct way 
to get more flow from 0 to 5, but if we change the switch at 1 to redirect enough flow 
to fill l->4, we open up enough capacity in 3->5 to allow us to add a unit of flow on 
0->2->3->5. Even for this simple network, finding switch settings that increase the flow 
is not an easy task; for a complicated network, we are clearly interested in the following 
question: What switch settings will maximize the amount of oil flowing from source 
to sink? We can model this situation directly with an edge-weighted digraph that has a 
single source and a single sink. The edges in the network correspond to the oil pipes, 
the vertices correspond to the junctions with switches that control how much oil goes 
into each outgoing edge, and the weights on the edges correspond to the capacity of the 
pipes. We assume that the edges are directed, specifying that oil can flow in only one di¬ 
rection in each pipe. Each pipe has a certain amount of flow, which is less than or equal 
to its capacity, and every vertex satisfies the equilibrium condition that the flow in is 
equal to the flow out. This flow-network abstraction is a useful problem-solving model 
that applies directly to a variety of applications and indirectly to still more. We some¬ 
times appeal to the idea of oil flowing through pipes for intuitive support of basic ideas, 



tinyFN.txt 


standard drawing 


drawing with capacities 


drawing with flow 





flow representation 

0 1 

2.0 

2.0 

0 2 

3.0 

1.0 

1 3 

3.0 

2.0 

1 4 

1.0 

0.0 

2 3 

1.0 

0.0 

2 4 

1.0 

1.0 

3 5 

2.0 

2.0 

4 5 

3.0 

1.0 


flow value 
associated 
with 

each edge 


Anatomy of a network-flow problem 


www.it-ebooks.info 














888 


CONTEXT 


but our discussion applies equally well to goods moving through distribution chan¬ 
nels and to numerous other situations. As with our use of distance in shortest-paths 
algorithms, we are free to abandon any physical intuition when convenient because 
all the definitions, properties, and algorithms that we consider are based entirely on 
an abstract model that does not necessarily obey physical laws. Indeed, a prime reason 
for our interest in the network-flow model is that it allows us to solve numerous other 
problems through reduction, as we see in the next section. 

Definitions. Because of this broad applicability, it is worthwhile to consider precise 
statements of the terms and concepts that we have just informally introduced: 

Definition. A flow network is an edge-weighted digraph with positive edge weights 
(which we refer to as capacities). An st-flow network has two identified vertices, a 
source s and a sink t. 


We sometimes refer to edges as having infinite capacity or, equivalently, as being un¬ 
capacitated. That might mean that we do not compare flow against capacity for such 
edges, or we might use a sentinel value that is guaranteed to be larger than any flow 
value. We refer to the total flow into a vertex (the sum of the flows on its incoming 
edges) as the vertex’s inflow, the total flow out of a vertex (the sum of the flows on its 
outgoing edges) as the vertex’s outflow, and the difference between the two (inflow 
minus outflow) as the vertex’s netflow. To simplify the discussion, we also assume that 
there are no edges leaving t or entering s. 

Definition. An st-flow in an st-flow network is a set of nonnegative values associ¬ 
ated with each edge, which we refer to as edge flows. We say that a flow is feasible if it 
satisfies the condition that no edge’s flow is greater than that edge’s capacity and the 
local equilibrium condition that the every vertex’s netflow is zero (except s and t). 


We refer to the sink’s inflow as the st-flow value. We will see in proposition c that the 
value is also equal to the source’s outflow. With these definitions, the formal statement 
of our basic problem is straightforward: 

Maximum st-flow. Given an st-flow network, find an st-flow such that no other 
flow from s to t has a larger value. 

For brevity, we refer to such a flow as a maxflow and the problem of finding one in a 
network as the maxflow problem. In some applications, we might be content to know 
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just the maxflow value, but we generally want to know a flow (edge flow values) that 
achieves that value. 

APIs. The FlowEdge and FlowNetwork APIs shown on page 890 are straightforward 
extensions of APIs from chapter 3 . We will consider on page 896 an implementation 
of FI owEdge that is based on adding an instance variable containing the flow to our 
Wei ghtedEdge class from page 610. Flows have a direction, but we do not base FlowEdge 
on Wei ghtedDi rectedEdge because we work with a more general abstraction known as 
the residual network that is described below, and we need each edge to appear in the adja¬ 
cency lists of both its vertices to implement the residual network. The residual network 
allows us to both add and subtract flow and to test whether an edge is full to capacity 
(no more flow can be added) or empty (no flow can be subtracted). This abstraction is 
implemented via the the methods residualCapacityO and addResidual Flow() that 
we will consider later. The implementation of FI owNetwork is virtually identical to our 
EdgeWei ghtedG raph implementation on page 611, so we omit it. To simplify the file for¬ 
mat, we adopt the convention that the source is 0 and the sink is V— 1. These APIs leave 
a straightforward goal for maxflow 
algorithms: build a network, then 
assign values to the flow instance 
variables in the client’s edges that 
maximize flow through the net¬ 
work. Shown at right are client 
methods for certifying whether a 
flow is feasible. Typically, we might 
do such a check as the final action 
of a maxflow algorithm. 


Checking that a flow is feasible in a flow network 


private boolean local Eq(FlowNetwork G, int v) 

{ // Check local equilibrium at each vertex, 

double EPSILON = IE-11; 
double netflow = 0.0; 
for (FlowEdge e : G.adj(v)) 

if (v == e.fromO) netflow -= e.flowO; 
else netflow += e.flowO; 

return Math.abs(netflow) < EPSILON; 

} 

private boolean isFeasible(FlowNetwork G) 

{ 

// Check that flow on each edge is nonnegative 
// and not greater than capacity, 
for (int v = 0; v < G.V(); v++) 
for (FlowEdge e : G.adj(v)) 

if (e.flowO < 0 || e.flowO > e.capO) 
return false; 

// Check local equilibrium at each vertex, 
for (int v = 0; v < G.V(); v++) 

if (v !=s && v != t && llocalEq(v)) 
return false; 

return true; 

} 
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public class 

FlowEdge 



FlowEdge(int v, int w, double 

cap) 

int 

f romO 

vertex this edge points from 

int 

to() 

vertex this edge points to 

int 

other(int v) 

other endpoint 

double 

capacity() 

capacity of this edge 

double 

flowO 

flow in this edge 

double 

residualCapacityTo(int v) 

residual capicity toward v 

double 

addFlowTo(int v, double delta) 

add del ta flow toward v 

String 

toString() 

string representation 


API for edges in a flow network 


public class FlowNetwork 



FlowNetwork(int V) 

empty V -vertex flow network 


FlowNetwork(In in) 

construct from input stream 


int VO 

number of vertices 


int E() 

number of edges 


void addEdge(FlowEdge e) 

add e to this flow network 


Iterable<FlowEdge> adj(int v) edges pointing from v 

Iterable<FlowEdge> edges() all edges in this flow network 

String toStringQ string representation 


Flow network API 
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Ford-Fulkerson algorithm. An effective approach to solving max- 
flow problems was developed by L. R. Ford and D. R. Fulkerson in 
1962. It is a generic method for increasing flows incrementally along 
paths from source to sink that serves as the basis for a family of algo¬ 
rithms. It is known as the Ford-Fulkerson algorithm in the classical lit¬ 
erature; the more descriptive term augmenting-path algorithm is also 
widely used. Consider any directed path from source to sink through 
an st-flow network. Let x be the minimum of the unused capacities of 
the edges on the path. We can increase the network’s flow value by at 
least x by increasing the flow in all edges on the path by that amount. 
Iterating this action, we get a first attempt at computing flow in a 
network: find another path, increase the flow along that path, and 
continue until all paths from source to sink have at least one full edge 
(so that we can no longer increase flow in this way). This algorithm 
will compute the maxflow in some cases but will fall short in other 
cases. Our introductory example on page 886 is such an example. To im¬ 
prove the algorithm such that it always finds a maxflow, we consider 
a more general way to increase the flow, along a path from source to 
sink through the network’s underlying undirected graph. The edges 
on any such path are either forward edges, which go with the flow 
(when we traverse the path from source to sink, we traverse the edge 
from its source vertex to its destination vertex), or backward edges, 
which go against the flow (when we traverse the path from source 
to sink, we traverse the edge from its destination vertex to its source 
vertex). Now, for any path with no full forward edges and no empty 
backward edges, we can increase the amount of flow in the network 
by increasing flow in forward edges and decreasing flow in backward 
edges. The amount by which the flow can be increased is limited by 
the minimum of the unused capacities in the forward edges and the 
flows in the backward edges. Such a path is called an augmenting path. 
An example is shown at right. In the new flow, at least one of the 
forward edges along the path becomes full or at least one of the back¬ 
ward edges along the path becomes empty. The process just sketched 
is the basis for the classical Ford-Fulkerson maxflow algorithm (aug- 
menting-path method). We summarize it as follows: 






An augmenting path 
(0->2->3->l->4->5) 
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Ford-Fulkerson maxflow algorithm. Start with zero flow everywhere. Increase the 
flow along any augmenting path from source to sink (with no full forward edges or 
empty backward edges), continuing until there are no such paths in the network. 


Remarkably (under certain technical conditions about numeric properties of the flow), 
this method always finds a maxflow, no matter how we choose the paths. Like the 
greedy MST algorithm discussed in section 4.3 and the generic shortest-paths method 
discussed in section 4 . 4 , it is a generic algorithm that is useful because it establishes 
the correctness of a whole family of more specific algorithms. We are free to use any 
method whatever to choose the path. Several algorithms that compute sequences of 
augmenting paths have been developed, all of which lead to a maxflow. The algorithms 
differ in the number of augmenting paths they compute and the costs of finding each 
path, but they all implement the Ford-Fulkerson algorithm and find a maxflow. 

Maxflow-mincut theorem. To show that any flow computed by any implementation 
of the Ford-Fulkerson algorithm is indeed a maxflow, we prove a key fact known as the 
maxflow-mincut theorem. Understanding this theorem is a crucial step in understand¬ 
ing network-flow algorithms. As suggested by its name, the theorem is based on a direct 
relationship between flows and cuts in networks, so we begin by defining terms that 
relate to cuts. Recall from section 4.3 that a cut in a graph is a partition of the vertices 
into two disjoint sets, and a crossing edge is an edge that connects a vertex in one set to 
a vertex in the other set. For flow networks, we refine these definitions as follows: 


Definition. An st-cut is a cut that places vertex s in one of its sets and vertex t in 
the other. 


Each crossing edge corresponding to an st-cut is either an st-edge that goes from a ver¬ 
tex in the set containing s to a vertex in the set containing t, or a ts-edge that goes in 
the other direction. We sometimes refer to the set of crossing sf-edges as a cut set. The 
capacity of an st-cut in a flow network is the sum of the capacities of that cut’s st-edges, 
and the flow across an st-cut is the difference between the sum of the flows in that cut’s 
st-edges and the sum of the flows in that cut’s ts-edges. Removing all the st-edges (the 
cut set) in an st-cut of a network leaves no path from s to t, but adding any one of them 
back could create such a path. Cuts are the appropriate abstraction for many applica¬ 
tions. For our oil-flow model, a cut provides a way to completely stop the flow of oil 
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from the source to the sink. If we view the capacity of the cut as the cost of doing so, to 
stop the flow in the most economical manner is to solve the following problem: 

Minimum st-cut. Given an st-network, find an st-c ut such that the capacity of 
no other cut is smaller. For brevity, we refer to such a cut as a mincut and to the 
problem of finding one in a network as the mincut problem. 

The statement of the mincut problem includes no mention of flows, and these defini¬ 
tions might seem to digress from our discussion of the augmenting-path algorithm. 
On the surface, computing a mincut (a set of edges) seems easier than computing a 
maxflow (an assignment of weights to all the edges). On the contrary, the maxflow and 
mincut problems are intimately related. The augmenting-path method itself provides 
a proof. That proof rests on the following basic relationship between flows and cuts, 
which immediately gives a proof that local equilibrium in an st-flow implies global 
equilibrium as well (the first corollary) and an upper bound on the value of any st-flow 
(the second corollary): 


Proposition E. For any st-flow, the flow across each st-cut 
is equal to the value of the flow. 

Proof: Let C s be the vertex set containing s and C t the 
vertex set containing t. This fact follows immediately by 
induction on the size of C t . The property is true by defi¬ 
nition when C t is t and when a vertex is moved from C s to 
C f , local equilibrium at that vertex implies that the stated 
property is preserved. Any st-cut can be created by mov¬ 
ing vertices in this way. 



Corollary. The outflow from s is equal to the inflow to t (the value of the st-flow). 
Proof: LetC s be{s}. 


Corollary. No st-flow’s value can exceed the capacity of any st-cut. 
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Proposition F. (Maxflow-mincut theorem) Let/be an st-flow. The following three 
conditions are equivalent: 

i. There exists an sf-cut whose capacity equals the value of the flow/. 

ii. f is a maxflow. 

iii. There is no augmenting path with respect to / 

Proof: Condition i. implies condition ii. by the corollary to proposition e. Con¬ 
dition ii. implies condition iii. because the existence of an augmenting path implies 
the existence of a flow with a larger flow value, contradicting the maximality off. 

It remains to prove that condition iii. implies condition i. Let C s be the set of all 
vertices that can be reached from s with an undirected path that does not contain a 
full forward or empty backward edge, and let C t be the remaining vertices. Then, t 
must be in C t , so (C s , C t ) is an st-cut, whose cut set consists entirely of full forward 
or empty backward edges. The flow across this cut is equal to the cut’s capacity 
(since forward edges are full and the backward edges are empty) and also to the 
value of the network flow (by proposition e). 


Corollary. (Integrality property) When capacities are integers, there exists an inte¬ 
ger-valued maxflow, and the Ford-Fulkerson algorithm finds it. 

Proof: Each augmenting path increases the flow by a positive integer (the mini¬ 
mum of the unused capacities in the forward edges and the flows in the backward 
edges, all of which are always positive integers). 


It is possible to design a maxflow with noninteger flows, even when capacities are all 
integers, but we do not need to consider such flows. From a theoretical standpoint, this 
observation is important: allowing capacities and flows that are real numbers, as we 
have done and as is common in practice, can lead to unpleasant anomalous situations. 
For example, it is known that the Ford-Fulkerson algorithm could, in principle, lead to 
an infinite sequence of augmenting paths that does not even converge to the maxflow 
value. The version of the algorithm that we consider is known to always converge, even 
when capacities and flows are real-valued. No matter what method we choose to find an 
augmenting path and no matter what paths we find, we always end up with a flow that 
does not admit an augmenting path, which therefore must be a maxflow. 
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Residual network. The generic Ford-Fulkerson algorithm does not specify any par¬ 
ticular method for finding an augmenting path. How can we find a path with no full 
forward edges and no empty backward edges? To this end, we begin with the following 
definition: 

Definition. Given a sf-flow network and an sf-flow, the residual network for the 
flow has the same vertices as the original and one or two edges in the residual net¬ 
work for each edge in the original, defined as follows: For each edge e from v to w in 
the original, let f e be its flow and c e its capacity. If/ e is positive, include an edge w->v 
in the residual with capacity f e ; and if f e is less than c e , include an edge v->w in the 
residual with capacity c e —f e . 

If an edge e from v to w is empty ({, is equal to 0), there is a single corresponding edge 
v->w with capacity c e in the residual; if it is full (f c is equal to c e ), there is a single cor¬ 
responding edge w->v with capacity^, in the residual; and if it is neither empty nor full, 
both v->w and w->v are in the residual with their respective capacities. An example is 
shown at the bottom of this page. At first, the residual network representation is a bit 
confusing because the edges corresponding to flow go in the opposite direction of the 
flow itself. The forward edges represent the remaining capacity (the amount of flow we 
can add if traversing that edge); the backward edges represent the flow (the amount of 
flow we can remove if traversing that edge). The code on page 896 gives the methods in the 
FI owEdge class that we need to implement the residual network abstraction. With these 
implementations, our algorithms work with the residual network, but they are actually 
examining capacities and changing flow (through edge references) in the client’s edg¬ 
es. The methods f rom() and other() allow us to process edges in either orientation: 


drawing with flow 


flow representation 


residual network 



(remaining capacity ) 


backward edge 
{flow) 


Anatomy of a network-flow problem (revisited) 
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Flow edge data type (residual network) 


public class FlowEdge 
{ 

private final int v; 

// edge source 

private final int w; 

// edge target 

private final double capacity; 

// capacity 

private double flow; 

// flow 


public FlowEdge(int v, int w, double capacity) 

{ 

this.v = v; 
this.w = w; 

this.capacity = capacity; 
this.flow = 0.0; 

} 


public 

int from() 

{ 

return 

v; 

} 

pub!ic 

int to() 

{ 

return 

w; 

} 

public 

double capacityO 

{ 

return 

capacity; 

} 

public 

double flow() 

{ 

return 

flow; 

} 


public int other(int vertex) 

// same as for Edge 

public double residualCapacityTo(int vertex) 

{ 

if (vertex == v) return flow; 

else if (vertex == w) return cap - flow; 

else throw new RuntimeException("Inconsistent edge"); 

} 

public void addResidualFlowTo(int vertex, double delta) 

{ 

if (vertex == v) flow -= delta; 

else if (vertex == w) flow += delta; 

else throw new RuntimeException("Inconsistent edge"); 

} 

public String toStringO 

{ return String.format("%d->%d %.2f %.2f", v, w, capacity, flow); } 


This FlowEdge implementation adds to the weighted Di rectedEdge implementation of section 4.4 
(see page 642) a f 1 ow instance variable and two methods to implement the residual flow network. 
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e.other(v) returns the endpoint of e that is not v. The methods residualCapToO 
and addResi dual FI owTo() implement the residual network. Residual networks allow 
us to use graph search to find an augmenting path, since any path from source to sink 
in the residual network corresponds directly to an augmenting path in the original 
network. Increasing the flow along the path implies making changes in the residual 
network: for example, at least one edge on the path becomes full or empty, so at least 
one edge in the residual network changes direction or disappears (but our use of an ab¬ 
stract residual network means that we just check for positive capacity and do not need 
to actually insert and delete edges). 

Shortest-augmenting-path method. Perhaps the simplest Ford-Fulkerson implemen¬ 
tation is to use a shortest augmenting path (as measured by the number of edges on the 
path, not flow or capacity). This method was suggested by J. Edmonds and R. Karp in 1972. 
In this case, the search for an augmenting path amounts to breadth-first search (BFS) 
in the residual network, precisely as described in section 4 . 1 , as you can see by com¬ 
paring the hasAugmenti ngPathO implementation below to our breadth-first search 
implemention in algorithm 4.2 on page 540 (the residual graph is a digraph, and this is 
fundamentally a digraph processing algorithm, as mentioned on page 685). This method 
forms the basis 
for the full im¬ 
plementation 
in ALGORITHM 
6.14 on the next 
page, a remark¬ 
ably concise 
implementation 
based on the tools 
we have devel¬ 
oped. For brevity, 
we refer to this 
method as the 
shortest-augment- 
ing-path maxflow 
algorithm. A trace 
for our example 
is shown in detail 
on page 899. 

Finding an augmenting path in the residual network via breadth-first search 


private boolean hasAugmentingPath(FIowNetwork C, int s, int t) 

{ 

marked = new boolean[G.V()]; // Is path to this vertex known? 

edgeTo = new FlowEdge[G.V()]; // last edge on path 
Queue<Integer> q = new Queue<Integer>(); 

marked[s] = true; // Mark the source 

q.enqueue(s); // and put it on the queue, 

while (Iq.isEmptyO) 

{ 

int v = q . dequeueO ; 

for (FlowEdge e : G.adj(v)) 

{ 

int w = e.other(v); 

if (e.residualCapacityTo(w) > 0 && !marked[w]) 

{ // For every edge to an unmarked vertex (in residual) 

edgeTo[w] = e; // Save the last edge on a path, 

marked[w] = true; // Mark w because a path is known 

q.enqueue(w); // and add it to the queue. 

} 

} 

} 

return marked[t]; 

} 
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ALGORITHM 6.14 Ford-Fulkerson shortest-augmenting path maxflow algorithm 


public class FordFulkerson 

{ 

private boolean[] marked; // Is s->v path in residual graph? 

private FlowEdge[] edgeTo; // last edge on shortest s->v path 
private double value; // current value of maxflow 

public FordFulkerson(FlowNetwork G, int s, int t) 

{ // Find maxflow in flow network G from s to t. 

while (hasAugmentingPath(G, s, t)) 

{ // While there exists an augmenting path, use it. 

// Compute bottleneck capacity. 

double bottle = Double.POSITIVE_INFINITY; 
for (int v = t; v != s; v = edgeTo[v].other(v)) 

bottle = Math.min(bottle, edgeTo[v].residualCapacityTo(v)); 

// Augment flow. 

for (int v = t; v != s; v = edgeTo[v].other(v)) 
edgeTo[v].addResidualFlowTo(v, bottle); 

value += bottle; 

} 

} 

public double value() { return value; } 

public boolean inCut(int v) { return marked[v]; } 

public static void main(String[] args) 

{ 

FlowNetwork G = new FlowNetwork(new ln(args[0])); 
int s = 0, t = G.V() - 1; 

FordFulkerson maxflow = new FordFulkerson(G, s, t); 

StdOut.println("Max flow from " + s + " to " + t); 
for (int v = 0; v < G.V(); v++) 
for (FlowEdge e : G.adj(v)) 

if ((v == e.fromO) && e.flow() > 0) 

StdOut.println(" " + e); 

StdOut. println("Max flow value = " + maxflow.valueO) ; 

} 

} 


This implementation of the Ford-Fulkerson algorithm finds the shortest augmenting path in the 
residual network, finds the bottneck capacity in that path, and augments the flow along that path, 
continuing until no path from source to sink exists. 
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initial empty network 



add 2 units of flow 




add 1 unit of flow 
along 0->2->3->l->4->5 


residual network 




st-cut 



Trace of augmenting-path Ford-Fulkerson algorithm 


% java FordFulkerson tinyFN.txt 
Max flow from 0 to 5 
0->2 3.0 2.0 
0->l 2.0 2.0 

l->4 1.0 1.0 

1- >3 B.O 1.0 

2- >3 1.0 1.0 

2- >4 1.0 1.0 

3- >5 2.0 2.0 

4- >5 3.0 2.0 

Max flow value = 4.0 
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Shortest augmenting paths in a larger flow network 


Performance. A larger example is shown in the figure above. As is evident from the 
figure, the lengths of the augmenting paths form a nondecreasing sequence. This fact 
is a first key to analyzing the performance of the algorithm. 

Proposition G. The number of augmenting paths needed in the shortest-augment- 
ing-path implementation of the Ford-Fulkerson maxflow algorithm for a flow net¬ 
work with V vertices and E edges is at most EV12. 

Proof sketch: Every augmenting path has a critical edge —an edge that is deleted 
from the residual network because it corresponds either to a forward edge that be¬ 
comes filled to capacity or a backward edge that is emptied. Each time an edge is a 
critical edge, the length of the augmenting path through it must increase by 2 (see 
exercise 6 . 39 ). Since an augmenting path is of length at most V each edge can be 
on at most V/2 augmenting paths, and the total number of augmenting paths is at 
most EVI2. 
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Corollary. The shortest-augmenting-path implementation of the Ford-Fulkerson 
maxflow algorithm takes time proportional to VE 2 /2 in the worst case. 

Proof: Breadth-first search examines at most E edges. 


The upper bound of proposition g is very conservative. For example, the graph shown 
in the figure at the top of page 900 has 11 vertices and 20 vertices, so the bound says that 
the algorithm uses no more than 110 augmenting paths. In fact, it uses 14. 

Other implementations. Another Ford-Fulkerson implementation, suggested by Ed¬ 
monds and Karp, is the following: Augment along the path that increases the flow by 
the largest amount. For brevity, we refer to this method as the maximum-capacity- 
augmenting-path maxflow algorithm. We can implement this (and other approaches) 
by using a priority queue and slightly modifying our implementation of Dijkstra’s 
shortest-paths algorithm, choosing edges from the priority queue to give the maximum 
amount of flow that can be pushed through a forward edge or diverted from a back¬ 
ward edge. Or, we might look for a longest augmenting path, or make a random choice. 
A complete analysis establishing which method is best is a complex task, because their 
running times depend on 

■ The number of augmenting paths needed to find a maxflow 

■ The time needed to find each augmenting path 

These quantities can vary widely, depending on the network being processed and on 
the graph-search strategy. Several other approaches to solving the maxflow problem 
have also been devised, some of which compete well with the Ford-Fulkerson algo¬ 
rithm in practice. Developing a mathematical model of maxflow algorithms that can 
validate such hypotheses, however, is a significant challenge. The analysis of maxflow 
algorithms remains an interesting and active area of research. From a theoretical stand¬ 
point, worst-case performance bounds for numerous maxflow algorithms have been 
developed, but the bounds are generally substantially higher than the actual costs ob¬ 
served in applications and also quite a bit higher than the trivial (linear-time) lower 
bound. This gap between what is known and what is possible is larger than for any 
other problem that we have considered (so far) in this book. 
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the practical application of maxflow algorithms remains both an art and a science. 
The art lies in picking the strategy that is most effective for a given practical situation; 
the science lies in understanding the essential nature of the problem. Are there new data 
structLires and algorithms that can solve the maxflow problem in linear time, or can we 
prove that none exist? 


algorithm 


worst-case order of growth 
of running time 
for V vertices and E edges 
with integral capacities (max C) 


Ford-Fulkerson 


VE 2 


shortest augmenting path 


Ford-Fulkerson 


E 2 log C 


maximal augmenting path 


preflow-push 
possible ? 


EV log (£/ V 2 ) 
V +£? 


Performance characteristics of maxflow algorithms 
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Reduction Throughout this book, we have focused on articulating specific prob¬ 
lems, then developing algorithms and data structures to solve them. In several cases 
(many of which are listed below), we have found it convenient to solve a problem by 
formulating it as an instance of another problem that we have already solved. Formal¬ 
izing this notion is a worthwhile starting point for studying relationships among the 
diverse problems and algorithms that we have studied. 

Definition. We say that a problem A reduces to another problem B if we can use an 
algorithm that solves B to develop an algorithm that solves A. 


This concept is certainly a familiar one in software development: when you use a library 
method to solve a problem, you are reducing your problem to the one solved by the li¬ 
brary method. In this book, we have informally referred to problems that we can reduce 
to a given problem as applications. 

Sorting reductions. We first encountered reduction in chapter 2, to express the idea 
that an efficient sorting algorithm is useful for efficiently solving many other problems, 
that may not seem to be at all related to sorting. For example, we considered the follow¬ 
ing problems, among many others: 

Finding the median. Given a set of numbers, find the median value. 

Distinct values. Determine the number of distinct values in a set of numbers. 

Scheduling to minimize average completion time. Given a set of jobs of speci¬ 
fied duration to be completed, how can we schedule the jobs on a single processor 
so as to minimize their average completion time? 


Proposition H. The following problems reduce to sorting: 

■ Finding the median 

■ Counting distinct values 

■ Scheduling to minimize average completion time 
Proof: See page 345 and exercise 2 . 5 . 12 . 


Now, we have to pay attention to cost when doing a reduction. For example, we can find 
the median of a set of numbers in linear time, but using the reduction to sorting will 
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end up costing linearithmic time. Even so, such extra cost might be acceptable, since we 
can use an exisiting sort implementation. Sorting is valuable for three reasons: 

■ It is useful in its own right. 

■ We have an efficient algorithms for solving it. 

■ Many problems reduce to it. 

Generally, we refer to a problem with these properties as a problem-solving model. Like 
well-engineered software libraries, well-designed problem-solving models can greatly 
expand the universe of problems that we can efficiently address. One pitfall in focusing 
on problem-solving models is known as Maslow’s hammer, an idea widely attributed 
to A. Maslow in the 1960s: If all you have is a hammer, everything seems to be a nail. By 
focusing on a few problem-solving models, we may use them like Maslow’s hammer 
to solve every problem that comes along, depriving ourselves of the opportunity to 
discover better algorithms to solve the problem, or even new problem-solving models. 
While the models we consider are important, powerful, and broadly useful, it is also 
wise to consider other possibilities. 

Shortest-paths reductions. In section 4 . 4 , we revisited the idea of reduction in the 
context of shortest-paths algorithms. We considered the following problems, among 
many others: 

Single-source shortest paths in undirected graphs. Given an edge-weighted un¬ 
directed graph with nonnegative weights and a source vertex s, support queries of 
the form Is there a path from s to a given target vertex v? If so, find a shortest such 
path (one whose total weight is minimal). 

Parallel precedence-constrained scheduling. Given a set of jobs of specified du¬ 
ration to be completed, with precedence constraints that specify that certain jobs 
have to be completed before certain other jobs are begun, how can we schedule 
the jobs on identical processors (as many as needed) such that they are all com¬ 
pleted in the minimum amount of time while still respecting the constraints? 

Arbitrage. Find an arbitrage opportunity in a given table of currency-conversion 
rates. 

Again, the latter two problems do not seem to be directly related to shortest-paths 
problems, but we saw that shortest paths is an effective way to address them. These ex¬ 
amples, while important, are merely indicative. A large number of important problems, 
too many to survey here, are known to reduce to shortest paths—it is an effective and 
important problem-solving model. 
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Proposition I. The following problems reduce to shortest paths in weighted 
digraphs: 

■ Single-source shortest paths in undirected graphs with nonnegative weights 

■ Parallel precedence-constrained scheduling 

■ Arbitrage 

■ [many other problems] 

Proof examples: See page 654, page 665, and page 680. 


Maxflow reductions. Maxflow algorithms are also important in a broad context. We 
can remove various restrictions on the flow network and solve related flow problems; we 
can solve other network- and graph-processing problems; and we can solve problems 
that are not network problems at all. For example, consider the following problems. 

Job placement. A college’s job-placement office arranges interviews for a set of 
students with a set of companies; these interviews result in a set of job offers. As¬ 
suming that an interview followed by a job offer represents mutual interest in the 
student taking a job at the company, it is in everyone’s best interests to maximize 
the number of job placements. Is it possible to match every student with a job? 
What is the maximum number of jobs that can be filled? 

Product distribution. A company that manufactures a single product has fac¬ 
tories, where the product is produced; distribution centers, where the product is 
stored temporarily; and retail outlets, where the product is sold. The company 
must distribute the product from factories through distribution centers to retail 
outlets on a regular basis, using distribution channels that have varying capacities. 

Is it possible to get the product from the warehouses to the retail outlets such that 
supply meets demand everywhere? 

Network reliability. A simplified model considers a computer network as con¬ 
sisting of a set of trunk lines that connect computers through switches such that 
there is the possibility of a switched path through trunk lines connecting any two 
given computers. What is the minimum number of trunk lines that can be cut to 
disconnect some pair of computers? 

Again, these problems seem to be unrelated to one another and to flow networks, but 
they all reduce to maxflow. 
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Proposition J. The following problems reduce to the maxflow problem: 

■ Job placement 

■ Product distribution 

■ Network reliability 

■ [many other problems] 

Proof example: We prove the first (which is known as the maximum bipartite 
matching problem) and leave the others for exercises. Given a job-placement prob¬ 
lem, construct an instance of the maxflow problem by directing all edges from 
students to companies, adding a source vertex with edges directed to all the stu¬ 
dents and adding a sink vertex with edges directed from all the companies. Assign 
each edge capacity 1. Now, any integral solution to the maxflow problem for this 
network provides a solution to the corresponding bipartite matching problem (see 
the corollary to proposition f). The matching corresponds exactly to those edges 
between vertices in the two sets that are filled to capacity by the maxflow algorithm. 
First, the network flow always gives a legal matching: since each vertex has an edge 
of capacity 1 either coming in (from the sink) or going out (to the source), at most 
1 unit of flow can go through each vertex, implying in turn that each vertex will be 
included at most once in the matching. Second, no matching can have more edges, 
since any such matching would lead directly to a better flow than that produced by 
the maxflow algorithm. 


bipartite matching problem 


matching (solution) 


1 

Alice 

7 

Adobe 


Adobe 


Alice 


Amazon 


Bob 


Facebook 


Dave 

2 

Bob 

8 

Amazon 


Adobe 


Alice 


Amazon 


Bob 


Yahoo 


Dave 

3 

Carol 

9 

Facebook 


Facebook 


Alice 


Google 


Carol 


IBM 

10 Google 

4 

Dave 


Carol 


Adobe 


Eliza 


Amazon 

11 

IBM 

5 

Eliza 


Carol 


Google 


Eliza 


IBM 


Frank 


Yahoo 

12 Yahoo 

6 

Frank 


Bob 


IBM 


Eliza 


Yahoo 


Frank 


network-flow formulation 


maximum flow 



Example of reducing maximum bipartite matching to network flow 


Alice — Amazon 
Bob — Yahoo 
Carol — Facebook 
Dave — Adobe 
Eliza — Google 
Frank — IBM 
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For example, as illustrated in the figure at right, an augmenting-path max- 
flow algorithm might use the paths s->l->7->t,s->2->8->t,s->3->9->t, 
s->5->10->t, s->6->ll->t, and s->4->7->l->8->2->12->t to compute 
the matching 1-8, 2-12, 3-9, 4-7, 5-10, and 6-11. Thus, there is a way to 
match all the students to jobs in our example. Each augmenting path fills 
one edge from the source and one edge into the sink. Note that these edges 
are never used as back edges, so there are at most V” augmenting paths, and 
a total running time proportional to VE. 

SHORTEST PATHS AND MAXFLOW ARE IMPORTANT problem-solving models 
because they have the same properties that we articulated for sorting: 

■ They are useful in their own right. 

■ We have efficient algorithms for solving them. 

■ Many problems reduce to them. 

This short discussion serves only to introduce the idea. If you take a course 
in operations research, you will learn many other problems that reduce to 
these and many other problem-solving models. 

Linear programming. One of the cornerstones of operations research is 
linear programming (LP). It refers to the idea of reducing a given problem 
to the following mathematical formulation: 

Linear programming. Given a set of M linear inequalities and linear 
equations involving N variables, and a linear objective function of the N 
variables, find an assignment of values to the variables that maximizes 
the objective function, or report that no feasible assignment exists. 

Linear programming is an extremely important prob¬ 
lem-solving model because 

■ A great many important problems reduce to 
linear programming 

■ We have efficient algorithms for solving linear- 
programming problems 

The “useful in its own right” phrase is not needed 
in this litany that we have stated for other problem¬ 
solving models because so many practical problems 
reduce to linear programming. 


Maximize f + h 
subject to the constraints 

0 £ a £ 2 
0 £ b £ 3 
0£ c£3 
0 £ 4 £ 1 
0 £ e £ 1 
0 £/£ 1 
0£g£2 

0 £ /: £ 3 
a = c+d 
b = e+f 
c+e = g 

d+f= h 

LP example 







Augmenting paths for 
bipartite matching 
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Proposition K. The following problems reduce to linear programming 

■ Maxflow 

■ Shortest paths 

■ [many, many other problems] 

Proof example: We prove the first and leave the second to exercise 649 . We con¬ 
sider a system of inequalities and equations that involve one variable correspond¬ 
ing to each edge, two inequalities corresponding to each edge, and one equation 
corresponding to each vertex (except the source and the sink). The value of the 
variable is the edge flow, the inequalities specify that the edge flow must be between 
0 and the edge’s capacity, and the equations specify that the total flow on the edges 
that go into each vertex must be equal to the total flow on the edges that go out of 
that vertex. Any max flow problem can be converted into an instance of a linear 
programming problem in this way, and the solution is easily converted to a solution 
of the maxflow problem. The illustration below gives the details for our example. 


maxflow problem 

V~. 


2.0 

3.0 

3.0 

1.0 

1.0 

1.0 

2.0 

3.0 

t 

capacities 



LP formulation 

Maximize x 35 +x 45 
subject to the constraints 


0 — *01 
0 — *02 
0 — *13 
0 — *14 

0 £ x 23 
0 £x 24 
0£x 35 


2 

3 
3 

1 

1 

1 

2 

3 

*14 

*24 

*35 

*45 


LP solution 


1=2 

2=2 

3=1 

4=1 

3=1 

4=1 

5=2 

4=2 


maxflow solution 

Max flow from 0 to 5 
0->2 3.0 2.0 
0->l 2.0 2.0 
l->4 1.0 1.0 

1- >3 3.0 1.0 

2- >3 1.0 1.0 

2- >4 1.0 1.0 

3- >5 2.0 2.0 

4- >5 3.0 2.0 

Max flow value: 4.0 



Example of reducing network flow to linear programming 
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The “many, many other problems” in the statement of proposition k refers to three 
ideas. First, it is very easy to extend a model and to add constraints. Second, reduction 
is transitive, so all the problems that reduce to shortest paths and maximum flow also 
reduce to linear programming. Third, and more generally, optimization problems of all 
sorts can be directly formulated as linear programming problems. Indeed, the term lin¬ 
ear programming means “formulate an optimization problem as a linear programming 
problem.” This use predates the use of the word programming for computers. Equally 
important as the idea that a great many problems reduce to linear programming is the 
fact that efficient algorithms have been known for linear programming for many de¬ 
cades. The most famous, developed by G. Dantzig in the 1940s, is known as the simplex 
algorithm. Simplex is not difficult to understand (see the bare-bones implementation 
on the booksite). More recently, the ellipsoid algorithm presented by L. G. Khachian in 
1979 led to the development of interior point methods in the 1980s that have proven to 
be an effective complement to the simplex algorithm for the huge linear programming 
problems that people are solving in modern applications. Nowadays, linear program¬ 
ming solvers are robust, extensively tested, efficient, and critical to the basic operation 
of modern corporations. Uses in scientific contexts and even in applications program¬ 
ming are also greatly expanding. If you can model your problem as a linear program¬ 
ming problem, you are likely to be able to solve it. 

IN A VERY REAL SENSE, LINEAR PROGRAMMING IS THE PARENT of problem-solving 
models, since so many problems reduce to it. Naturally, this idea leads to the question 
of whether there is an even more powerful problem-solving model than linear pro¬ 
gramming. What sorts of problems do not reduce to linear programming? Here is an 
example of such a problem: 

Load balancing. Given a set of jobs of specified duration to be completed, how 
can we schedule the jobs on two identical processors so as to minimize the com¬ 
pletion time of all the jobs? 

Can we articLilate a more general problem-solving model and solve instances of prob¬ 
lems within that model efficiently? This line of thinking leads to the idea of intractabil¬ 
ity, our last topic. 
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Intractability The algorithms that we have studied in this book generally are used 
to solve practical problems and therefore consume reasonable amounts of resources. 
The practical utility of most of the algorithms is obvious, and for many problems we 
have the luxury of several efficient algorithms to choose from. Unfortunately, many 
other problems arise in practice that do not admit such efficient solutions. What’s 
worse, for a large class of such problems we cannot even tell whether or not an effi¬ 
cient solution exists. This state of affairs has been a source of extreme frustration for 
programmers and algorithm designers, who cannot find any efficient algorithm for a 
wide range of practical problems, and for theoreticians, who have been unable to find 
any proof that these problems are difficult. A great deal of research has been done in 
this area and has led to the development of mechanisms by which new problems can 
be classified as being “hard to solve” in a particular technical sense. Though much of 
this work is beyond the scope of this book, the central ideas are not difficult to learn. 
We introduce them here because every programmer, when faced with a new problem, 
should have some understanding of the possibility that there exist problems for which 
no one knows any algorithm that is guaranteed to be efficient. 

Groundwork. One of the most beautiful and intriguing intellectual discoveries of 
the 20th century, developed by A. Turing in the 1930s, is the Turing machine, a simple 
model of computation that is general enough to embody any computer program or 
computing device. A Turing machine is a finite-state machine that can read inputs, 
move from state to state, and write outputs. Turing machines form the foundation of 
theoretical computer science, starting with the following two ideas: 

■ Universality. All physically realizable computing devices can be simulated by a 
Turing machine. This idea is known as the Church-Turing thesis. This is a state¬ 
ment about the natural world and cannot be proven (but it can be falsified). The 
evidence in favor of the thesis is that mathematicians and computer scientists 
have developed numerous models of computation, but they all have been proven 
equivalent to the Turing machine. 

■ Computability. There exist problems that cannot be solved by a Turing machine 
(or by any other computing device, by universality). This is a mathematical 
truth. The halting problem (no program can guarantee to determine whether a 
given program will halt) is a famous example of such a problem. 

In the present context, we are interested in a third idea, which speaks to the efficiency 
of computing devices: 

■ Extended Church-Turing thesis. The order of growth of the running time of a 
program to solve a problem on any computing device is within a polynomial 
factor of some program to solve the problem on a Turing machine (or any other 
computing device). 
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Again, this is a statement about the natural world, buttressed by the idea that all known 
computing devices can be simulated by a Turing machine, with at most a polynomial 
factor increase in cost. In recent years, the idea of quantum computing has given some 
researchers reason to doubt the extended Church-Turing thesis. Most agree that, from a 
practical point of view, it is probably safe for some time, but many researchers are hard 
at work on trying to falsify the thesis. 


Exponential running time. The purpose of the theory of intractability is to separate 
problems that can be solved in polynomial time from problems that (probably) require 
exponential time to solve in the worst case. It is useful to think of an exponential-time 
algorithm as one that, for some input of size N, takes time proportional to 2 N (at least). 
The substance of the argument does not change if we replace 2 by any number a> 1. 
We generally take as granted that an exponential-time algorithm cannot be guaranteed 
to solve a problem of size 100 (say) in a reasonable amount of time, because no one 
can wait for an algorithm to take 2 100 steps, regardless of the speed of the computer. 
Exponential growth dwarfs technological changes: a supercomputer may be a trillion 
times faster than an abacus, but neither can come close to solving a problem that re¬ 
quires 2 100 steps. Sometimes the 
line between “easy” and “hard” 
problems is a fine one. For ex¬ 
ample, we studied an algorithm 
in section 4.1 that can solve the 
following problem: 


Shortest-path length. What 
is the length of the shortest 
path from a given vertex s to a 
given vertex tin a given graph? 

But we did not study algorithms 
for the following problem, which 
seems to be virtually the same: 

Longest-path length. What 
is the length of the longest 
simple path from a given ver¬ 
tex s to a given vertex t in a 
given graph? 


public class LongestPath 
{ 

private boolean[] marked; 
private int max; 

public LongestPath(Graph G, int s, int t) 

{ 

marked = new boolean[G.V()]; 
dfs(G, s, t, 0); 

} 

private void dfs(Graph G, int v, int t, int i) 
{ 

if (v == t && i > max) max = i ; 
if (v == t) return; 
marked[v] = true; 
for (int w : G.adj(v)) 

if (!marked[w]) dfs(G, w, t, i+1); 
marked [v] = false; 

} 

public int maxl_ength() 

{ return max; } 


Finding the length of the longest path in a graph 
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The crux of the matter is this: as far as we know, these problems are nearly at opposite 
ends of the spectrum with respect to difficulty. Breadth-first search yields a solution for 
the first problem in linear time, but all known algorithms for the second problem take 
exponential time in the worst case. The code at the bottom of the previous page shows a 
variant of depth-first search that accomplishes the task. It is quite similar to depth-first 
search, but it examines all simple paths from s to tin the digraph to find the longest one. 

Search problems. The great disparity between problems that can be solved with “ef¬ 
ficient” algorithms of the type we have been studying in this book and problems where 
we need to look for a solution among a potentially huge number of possibilities makes 
it possible to study the interface between them with a simple formal model. The first 
step is to characterize the type of problem that we study: 

Definition. A search problem is a problem having solutions with the property that 
the time needed to certify that any solution is correct is bounded by a polynomial in 
the size of the input. We say that an algorithm solves a search problem if, given any 
input, it either produces a solution or reports that none exists. 


Four particular problems that are of interest in our discussion of intractability are 
shown at the top of the facing page. These problems are known as satisfiability prob¬ 
lems. Now, all that is required to establish that a problem is a search problem is to show 
that any solution is sufficiently well-characterized that you can efficiently certify that 
it is correct. Solving a search problem is like searching for a “needle in a haystack” with 
the sole proviso that you can recognize the needle when you see it. For example, if you 
are given an assignment of values to variables in each of the satisfiability problems at 
the top of page 913, you easily can certify that each equality or inequality is satisfied, but 
searching for such an assignment is a totally different task. The name NP is commonly 
used to describe search problems—we will describe the reason for the name on page 914: 

Definition. NP is the set of all search problems. 


NP is nothing more than a precise characterization of all the problems that scientists, 
engineers, and applications programmers aspire to solve with programs that are guar¬ 
anteed to finish in a feasible amount of time. 
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Linear equation satisfiability. Given a set of M linear equations involving N 
variables, find an assignment of values to the variables that satisfies all of the 
equations, or report that none exists. 

Linear inequality satisfiability (search formulation of linear program¬ 
ming). Given a set of M linear inequalities involving N variables, find an assign¬ 
ment of values to the variables that satisfies all of the inequalities, or report that 
none exists. 

0-1 integer linear inequality satisfiability (search formulation of 0-1 integer 
linear programming). Given a set of M linear inequalities involving N integer 
variables, find an assignment of the values 0 or 1 to the variables that satisfies all 
of the inequalities, or report that none exists. 

Boolean satisfiability. Given a set of M equations involving and and or opera¬ 
tions on N boolean variables, find an assignment of values to the variables that 
satisfies all of the equations, or report that none exists. 

Selected search problems 


Other types of problems. The concept of search problems is one of many ways to char¬ 
acterize the set of problems that form the basis of the study of intractability. Other 
possibilities are decision problems (does a solution exist?) and optimization problems 
(what is the best solution)? For example, the longest-paths length problem on page 911 
is an optimization problem, not a search problem (given a solution, we have no way 
to verify that it is a longest-path length). A search version of this problem is to find a 
simple path connecting all the vertices (this problem is known as the Hamiltonian path 
problem). A decision version of the problem is to ask whether there exists a simple path 
connecting all the vertices. Arbitrage, boolean satisfiability, and Hamiltonian path are 
search problems; to ask whether a solution exists to any of these problems is a decision 
problem; and shortest/longest paths, maxflow, and linear programming are all optimi¬ 
zation problems. While not technically equivalent, search, decision, and optimization 
problems typically reduce to one another (see exercise 6.58 and 6.59) and the main 
conclusions we draw apply to all three types of problems. 
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Easy search problems. The definition of NP says nothing about the difficulty of finding 
the solution, just certifying that it is a solution. The second of the two sets of problems 
that form the basis of the study of intractability, which is known as P, is concerned with 
the difficulty of finding the solution. In this model, the efficiency of an algorithm is a 
function of the number of bits used to encode the input. 

Definition. P is the set of all search problems that can be solved in polynomial time. 


Implicit in the definition is the idea that the polynomial time bound is a worst-case 
bound. For a problem to be in P, there must exist an algorithm that can guarantee to 
solve it in polynomial time. Note that the polynomial is not specified at all. Linear, lin- 
earithmic, quadratic, and cubic are all polynomial time bounds, so this definition cer¬ 
tainly covers the standard algorithms we have studied so far. The time taken by an algo¬ 
rithm depends on the computer used, but the extended Church-Turing thesis renders 
that point moot—it says that a polynomial-time solution on any computing device 
implies the existence of a polynomial-time solution on any other computing device. 
Sorting belongs to P because (for example) insertion sort runs in time proportional to 
N 2 (the existence of linearithmic sorting algorithms is not relevant in this context), as 
does shortest paths, linear equation satisfiability, and many others. Having an efficient 
algorithm to solve a problem is a proof that the problem is in P. In other words, P is 
nothing more than a precise characterization of all the problems that scientists, engi¬ 
neers, and applications programmers do solve with programs that are guaranteed to 
finish in a feasible amount of time. 

Nondeterminism. The N in NP stands for nondeterminism. It represents the idea that 
one way (in theory) to extend the power of a computer is to endow it with the power 
of nondeterminism: to assert that when an algorithm is faced with a choice of several 
options, it has the power to “guess” the right one. For the purposes of our discus¬ 
sion, we can think of an algorithm for a nondeterministic machine as “guessing” the 
solution to a problem, then certifying that the solution is valid. In a Turing machine, 
nondeterminism is as simple as defining two different successor states for a given state 
and a given input and characterizing solutions as all legal paths to the desired result. 
Nondeterminism may be a mathematical fiction, but it is a useful idea. For example, 
in section 5.4, we used nondeterminism as a tool for algorithm design—our regular 
expression pattern-matching algorithm is based on efficiently simulating a nondeter¬ 
ministic machine. 
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problem 

input 

description 

poly-time 

algorithm 


instance 


solution 









Hamiltonian 

path 

graph G 

find a simple path 
that visits every vertex 

? 

(2 

) 


0-2-1-3 

factoring 

integer x 

find a nontrivial 

factor of x 

? 

97605257271 

8784561 

0-1 linear 
inequality 
satisfiability 

MO-1 

variables 

N inequalities 

assign values to the 
variables that 
satisfy the inequalities 

? 

X- y< 1 
2 x — z £ 2 
x + y ^ 2 
z^O 


x = 1 

y = 1 

z= 0 

all problems 
in P 


see table below 








Examples of problems in NP 





problem 

input 

description 

poly-time 

algorithm 


instance 


solution 







Pi 


shortest st-path 

graph G 
vertices S, t 

find the shortest path 
from s to t 

BFS 


X x x 

2J \ 


0-3 

t 

sorting 

array a 

find a permutation 
that puts a in 
ascending order 

mergesort 

2.8 

8.5 4.1 

1.3 

3 0 2 1 

linear equation 
satisfiability 

M variables 
N equations 

assign values to the 
variables that 
satisfy the equations 

Gaussian 

elimination 

x + y- 1.5 

2 x — y - 0 

x = 0.5 

y = i 

linear 

inequality 

satisfiability 

M variables 
N inequalities 

assign values to the 
variables that 
satisfy the inequalities 

ellipsoid 

x — y s 1.5 

2 x — z ^ 0 
x + y S 3.5 
z ^ 4.0 

x =2.0 
y= 1.5 
z= 4.0 


Examples of problems in P 


www.it-ebooks.info 










916 


CONTEXT 


The main question. Nondeterminism is such a powerful notion that it seems almost 
absurd to consider it seriously. Why bother considering an imaginary tool that makes 
difficult problems seem trivial? The answer is that, powerful as nondeterminism may 
seem, no one has been able to prove that it helps for any particular problem! Put an¬ 
other way, no one has been able to find a single problem that can be proven to be in NP 
but not in P (or even prove that one exists), leaving the following question open: 

Does P = NP? 

This question was first posed in a famous letter from K. Godel to J. von Neumann in 
1950 and has completely stumped mathematicians and computer scientists ever since. 
Other ways of posing the question shed light on its fundamental nature: 

■ Are there any hard-to-solve search problems? 

■ Would we be able to solve some search problems more efficiently if we could 
build a nondeterministic computing device? 

Not knowing the answers to these questions is extremely frustrating because many im¬ 
portant practical problems belong to NP but may or may not belong to P (the best 
known deterministic algorithms could take exponential time). If we could prove that 
a problem does not belong to P, then we could abandon the search for an efficient so¬ 
lution to it. In the absence of such a proof, there is the possibility that some efficient 
algorithm has gone undiscovered. In fact, given the current state of our knowledge, 
there could be some efficient algorithm for every problem in NP, which would imply 
that many efficient algorithms have gone undiscovered. Virtually no one believes that 
P = NP, and a considerable amount of effort has gone into proving the contrary, but this 
remains the outstanding open research problem in computer science. 

Poly-time reductions. Recall from page 903 that we show that a problem A reduces to 
another problem B by demonstrating that we can solve any instance of A in three steps: 

■ Transform it to an instance of B. 

■ Solve that instance of B. 

■ Transform the solution of B to be a solution of A. 

As long as we can perform the transformations (and solve B ) efficiently, we can solve A 
efficiently. In the present context, for efficient we use the weakest conceivable definition: 
to solve A we solve at most a polynomial number of instances of B, using transforma¬ 
tions that require at most polynomial time. In this case, we say that A poly-time reduces 
to B. Before, we used reduction to introduce the idea of problem-solving models that 
can significantly expand the range of problems that we can solve with efficient algo¬ 
rithms. Now, we use reduction in another sense: to prove a problem to be hard to solve. If 
a problem A is known to be hard to solve, and A poly-time reduces to B, then B must be 
hard to solve, too. Otherwise, a guaranteed polynomial-time solution to B would give a 
guaranteed polynomial-time solution to A. 
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Proposition L. Boolean satisfiability poly-time reduces 
to 0-1 integer linear inequality satisfiability. 

Proof: Given an instance of boolean satisfiability, de¬ 
fine a set of inequalities with one 0-1 variable corre¬ 
sponding to each boolean variable and one 0-1 vari¬ 
able corresponding to each clause, as illustrated in 
the example at right. With this construction, we can 
tranform a solution to the integer 0-1 linear inequality 
satisfiability problem to a solution to the boolean sat¬ 
isfiability problem by assigning each boolean variable 
to be true if the corresponding integer variable is 1 and 
false if it is 0. 


boolean satisfiability problem 

(x'l or x 2 or xf) and 
(*! or x' 2 or x 3 ) and 
(jc'i or x' 2 or x'f) and 
(x\ or x' 2 or x 3 ) 


0-1 integer linear inequality satisfiability formulation 


Cjis I 

if and only if 
first clause is 
satisfiable 


c, > 1 - Xt 


Ci£(1-x 1 )+x 2 + x 3 


C2-*l 
C 2 s 1 - x 2 
C 2 - x 3 

c 2 == Xj + (l-x 2 ) + x 3 


Corollary. If satisfiability is hard to solve, then so is in¬ 
teger linear programming. 


C 3 >l-Xj 

c 3 >l -x 2 
c 3 >l-x 3 

r 3 — (1 - x 3 ) + l-* 2 + (l-* 3 ) 


This statement is a meaningful statement about the rela¬ 
tive difficulty of solving these two problems even in the 
absence of a precise definition of hard to solve. In the pres¬ 
ent context, by “hard to solve,” we mean “not in P.” We 
generally use the word intractable to refer to problems that 
are not in P. Starting with the seminal work of R. Karp in 
1972, researchers have shown literally tens of thousands 
of problems from a wide variety of applications areas to 
be related by reduction relationships of this sort. More¬ 
over, these relationships imply much more than just rela¬ 
tionships between the individual problems, a concept that 
we now address. 


C4>l-X! 
c 4— 1 " x 2 

c 4 >x 3 

C 4 <(1 -X[) + (1 -X 2 ) + x 3 


S — C 1 

s£c 2 s,s1 

-- if and only if 

5 — c 3 c’s are all 1 

sr<c 4 


s a Cj + c 2 + c 3 + c 4 - 3 


Example of reducing boolean satisfiability to 
0-1 integer linear inequality satisfiability 


NP-completeness. Many, many problems are known to belong to NP but probably do 
not belong to P. That is, we can easily certify that any given solution is valid, but, despite 
considerable effort, no one has been able to develop an efficient algorithm to find a so¬ 
lution. Remarkably, all of these many, many problems have an additional property that 
provides convincing evidence that P NP: 
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Definition. A search problem A is said to be NP-complete if all problems in NP poly¬ 
time reduce to A. 


This definition enables us to upgrade our definition of “hard to solve” to mean “intrac¬ 
table unless P = NP.” If any NP-complete problem can be solved in polynomial time on 
a deterministic machine, then so can all problems in NP (i.e., P = NP). That is, the collec¬ 
tive failure of all researchers to find efficient algorithms for all of these problems might 
be viewed as a collective failure to prove that P = NP. NP-complete problems, meaning 
that we do not expect to find guaranteed polynomial-time algorithms. Mosr practical 
search problems are known to be either in P or NP-complete. 

Cook-Levin theorem. Reduction uses the NP-completeness of one problem to imply 
the NP-completeness of another. But reduction cannot be used in one case: how was 
the first problem proven to be NP-complete? This was done independently by S. Cook 
and L. Levin in the early 1970s. 

Proposition M. (Cook-Levin theorem) Boolean satisfiability is NP-complete. 

Extremely brief proof sketch: The goal is to show that if there is a polynomial 
time algorithm for boolean satisfiability, then all problems in NP can be solved in 
polynomial time. Now, a nondeterministic Turing machine can solve any problem 
in NP, so the first step in the proof is to describe each feature of the machine in 
terms of logical formulas such as appear in the boolean satisfiability problem. This 
construction establishes a correspondence between every problem in NP (which 
can be expressed as a program on the nondeterministic Turing machine) and some 
instance of satisfiability (the translation of that program into a logical formula). 
Now, the solution to the satisfiability problem essentially corresponds to a simula¬ 
tion of the machine running the given program on the given input, so it produces a 
solution to an instance of the given problem. Further details of this proof are well 
beyond the scope of this book. Fortunately, only one such proof is really necessary: 
it is much easier to use reduction to prove NP-completeness. 


The Cook-Levin theorem, in conjunction with the thousands and thousands of poly¬ 
time reductions from NP-complete problems that have followed it, leaves us with two 
possible universes: either P = NP and no intractable search problems exist (all search 
problems can be solved in polynomial time); or P * NP, there do exist intractable search 
problems (some search problems cannot be solved in polynomial time). NP-complete 
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Two possible universes 


problems arise frequently in important natural practical ap¬ 
plications, so there has been strong motivation to find good 
algorithms to solve them. The fact that no good algorithm has 
been found for any of these problems is surely strong evidence 
that P * NP, and most researchers certainly believe this to be 
the case. On the other hand, the fact that no one has been able 
to prove that any of these problems do not belong to P could 
be construed to comprise a similar body of circumstantial evi¬ 
dence on the other side. Whether or not P = NP, the practical 
fact is that the best known algorithm for any of the NP-com¬ 
plete problems takes exponential time in the worst case. 


Classifying problems. To prove that a search problem is in P, we need to exhibit a 
polynomial-time algorithm for solving it, perhaps by reducing it to a problem known 
to be in P. To prove that a problem in NP is NP-complete, we need to show that some 
known NP-complete problem is poly-time reducible to it: that is, that a polynomial¬ 
time algorithm for the new problem could be used to solve the NP-complete problem, 
and then could, in turn, be used to solve all problems in NP. Thousands and thousands 
of problems have been shown to be NP-complete in this way, as we did for integer lin¬ 
ear programming in proposition l. The list on page 920, which includes several of the 
problems addressed by Karp, is representative, but contains only a tiny fraction of the 
known NP-complete problems. Classifying problems as being easy to solve (in P) or 
hard to solve (NP-complete) can be: 

■ Straightforward. For example, the venerable Gaussian elimination algorithm 
proves that linear equation satisfiability is in P. 

■ Tricky but not difficult. For example, developing a proof like the proof of propo¬ 
sition a takes some experience and practice, but it is easy to understand. 

■ Extremely challenging. For example, linear programming was long unclassified, 
but Khachian’s ellipsoid algorithm proves that linear programming is in P. 

■ Open. For example, graph isomorphism (given two graphs, find a way to rename 
the vertices of one to make it identical to the other) and factor (given an integer, 
find a nontrivial factor) are still unclassified. 

This is a rich and active area of current research, still involving thousands of research 
papers per year. As indicated by the last few entries on the list on page 920, all areas of sci¬ 
entific inquiry are affected. Recall that our definition of NP encompasses the problems 
that scientists, engineers, and applications programmers aspire to solve feasibly—all 
such problems certainly need to be classified! 
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Boolean satisfiability. Given a set of M equations involving N boolean variables, 
find an assignment of values to the variables that satisfies all of the equations, or 
report that none exists. 

Integer linear programming. Given a set of M linear inequalities involving N 
integer variables, find an assignment of values to the variables that satisfies all of 
the inequalities, or report that none exists. 

Load balancing. Given a set of jobs of specified duration to be completed and a 
time bound T, how can we schedule the jobs on two identical processors so as to 
complete them all by time T? 

Vertex cover. Given a graph and a integer C, find a set of C vertices such that each 
edge of the graph is incident to at least one vertex of the set. 

Hamiltonian path. Given a graph, find a simple path that visits each vertex ex¬ 
actly once, or report that none exists. 

Protein folding. Given energy level M, find a folded three-dimensional confor¬ 
mation of a protein having potential energy less than M. 

Ising model. Given an Ising model on a lattice of dimension three and an energy 
threshhold E, is there a subgraph with free energy less than E ? 

Risk portfolio of a given return. Given an investment portfolio with a given total 
cost, a given return, risk values assigned to each investment, and a threshold M, 
find a way to allocate the investments such that the risk is less than M. 


Some famous NP-complete problems 
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Coping with NP -completeness. Some sort of solution to this vast panopoly of prob¬ 
lems must be found in practice, so there is intense interest in finding ways to address 
them. It is impossible to do justice to this vast field of study in one paragraph, but we 
can briefly describe various approaches that have been tried. One approach is to change 
the problem and find an “approximation” algorithm that finds not the best solution but 
a solution guaranteed to be close to the best. For example, it is easy to find a solution 
to the Euclidean traveling salesperson problem that is within a factor of 2 of the opti¬ 
mal. Unfortunately, this approach is often not sufficient to ward off NP-completeness, 
when seeking improved approximations. Another approach is to develop an algorithm 
that solves efficiently virtually all of the instances that do arise in practice, even though 
there exist worst-case inputs for which finding a solution is infeasible. The most famous 
example of this approach are the integer linear programming solvers, which have been 
workhorses for many decades in solving huge optimizaiton problems in countless in¬ 
dustrial applications. Even though they could require exponential time, the inputs that 
arise in practice evidently are not worst-case inputs. A third approach is to work with 
“efficient” exponential algorithms, using a technique known as backtracking to avoid 
having to check all possible solutions. Finally, there is quite a large gap between polyno¬ 
mial and exponential time that is not addressed by the theory. What about an algorithm 
that runs in time proportional to N lo % N or 2' v ? 

all the applications areas we have studied in this book are touched by NP-com- 
pleteness: NP-complete problems arise in elementary programming, in sorting and 
searching, in graph processing, in string processing, in scientific computing, in systems 
programming, in operations research, and in any conceivable area where computing 
plays a role. The most important practical contribution of the theory of NP-complete¬ 
ness is that it provides a mechanism to discover whether a new problem from any of 
these diverse areas is “easy” or “hard.” If one can find an efficient algorithm to solve a 
new problem, then there is no difficulty. If not, a proof that the problem is NP-complete 
tells us that developing an efficient algorithm would be a stunning achievement (and 
suggests that a different approach should perhaps be tried). The scores of efficient al¬ 
gorithms that we have examined in this book are testimony that we have learned a 
great deal about efficient computational methods since Euclid, but the theory of NP- 
completeness shows that, indeed, we still have a great deal to learn. 
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EXERCISES on collision simulation 

6.1 Complete the implementation predictCollisionsC) and Particle as described 
in the text. There are three equations governing the elastic collision between a pair of 
hard discs: (a) conservation of linear momentum, ( b ) conservation of kinetic energy, 
and (c) upon collision, the normal force acts perpendicular to the surface at the colli¬ 
sion point (assuming no friction or spin). See the booksite for more details. 

6.2 Develop a version of Colli si onSystem, Particle, and Event that handles multi¬ 
particle collisions. Such collisions are important when simulating the break in a game 
of billiards. (This is a difficult exercise!) 

6.3 Develop a version of Col 1 i si onSystem, Parti cl e, and Event that works in three 
dimensions. 

6.4 Explore the idea of improving the performance of si mu 1 ate () in Col 1 i si onSystem 
by dividing the region into rectangular cells and adding a new event type so that you 
only need to predict collisions with particles in one of nine adjacent cells in any time 
quantum. This approach reduces the number of predictions to calculate at the cost of 
monitoring the movement of particles from cell to cell. 

6.5 Introduce the concept of entropy to Col 1 i si onSystem and use it to confirm clas¬ 
sical results. 

6.6 Brownian motion. In 1827, the botanist Robert Brown observed the motion 
of wildflower pollen grains immersed in water using a microscope. He observed that 
the pollen grains were in a random motion, following what would become known as 
Brownian motion. This phenomenon was discussed, but no convincing explanation 
was provided until Einstein provided a mathematical one in 1905. Einstein’s explana¬ 
tion: the motion of the pollen grain particles was caused by millions of tiny molecules 
colliding with the larger particles. Run a simulation that illustrates this phenomenon. 

6.7 Temperature. Add a method temperatureO to Parti cl e that returns the prod¬ 
uct of its mass and the square of the magitude of its velocity divided by dk B where d =2 is 
the dimension and k B =1.3806503 X 10" 23 is Boltzmann’s constant. The temperature of 
the system is the average value of these quantities. Then add a method temperatureO 
to Col 1 i si onSystem and write a driver that plots the temperature periodically, to check 
that it is constant. 
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6.8 Maxwell-Boltzmann. The distribution of velocity of particles in the hard disc 
model obeys the Maxwell-Boltzmann distribution (assuming that the system has ther- 
malized and particles are sufficiently heavy that we can discount quantum-mechanical 
effects), which is known as the Rayleigh distribution in two dimensions. The distribu¬ 
tion shape depends on temperature. Write a driver that computes a histogram of the 
particle velocities and test it for various temperatures. 

6.9 Arbitrary shape. Molecules travel very quickly (faster than a speeding jet) but dif¬ 
fuse slowly because they collide with other molecules, thereby changing their direction. 
Extend the model to have a boundary shape where two vessels are connected by a pipe 
containing two different types of particles. Run a simulation and measure the fraction 
of particles of each type in each vessel as a function of time. 

6.10 Rewind. After running a simulation, negate all velocities and then run the system 
backward. It should return to its original state! Measure roundoff error by measuring 
the difference between the final and original states of the system. 

6.11 Pressure. Add a method pressureO to Particle that measures pressure by 
accumulating the number and magnitude of collisions against walls. The pressure 
of the system is the sucm of these quantities. Then add a method pressureO to 
Col 1 i si onSystem and write a client that validates the equation pv = nRT. 

6.12 Index priority queue implementation. Develop a version of Colli si onSystem 
that uses an index priority queue to guarantee that the size of the priority queue is at 
most linear in the number of particles (instead of quadratic or worse). 

6.13 Priority queue performance. Instrument the priority queue and test Pressure 
at various temperatures to identify the computational bottleneck. If warranted, try 
switching to a different priority-queue implementation for better performance at high 
temperatures. 
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EXERCISES on B-Trees 

6.14 Suppose that, in a three-level tree, we can afford to keep a links in internal mem¬ 
ory, between b and 2b links in pages representing internal nodes, and between c and 2c 
items in pages representing external nodes. What is the maximum number of items that 
we can hold in such a tree, as a function of a, b, and c? 

6.15 Develop an implementation of Page that represents each B-tree node as a 
Bi narySearchST object. 

6.16 Extend BTreeSET to develop a BTreeST implementation that associates keys with 
values and supports our full ordered symbol table API that includes min(), max(), 
floorO, ceilingO, deleteMinO, deleteMaxO, selectO, rank(), and the two- 
argument versions of sizeO and get (). 

6.17 Write a program that uses StdDraw to visualize B-trees as they grow, as in the text. 

6.18 Estimate the average number of probes per search in a B-tree for S random 
searches, in a typical cache system, where the T most-recently-accessed pages are kept in 
memory (and therefore add 0 to the probe count). Assume that S is much larger than T. 

6.19 Web search. Develop an implementation of Page that represents B-tree nodes as 
text files on web pages, for the purposes of indexing (building a concordance for) the 
web. Use a file of search terms. Take web pages to be indexed from standard input. To 
keep control, take a command-line parameter m, and set an upper limit of 10 m internal 
nodes (check with your system administrator before running for large m). Use an m- 
digit number to name your internal nodes. For example, when m is 4, your nodes names 
might be BTreeNodeOOOO, BTreeNodeOOOl, BTreeNode0002, and so forth. Keep pairs 
of strings on pages. Add a cl ose () operation to the API, to sort and write. To test your 
implementation, ook for yourself and your friends on your university’s website. 

6.20 B* trees. Consider the sibling split (or B*-tree ) heuristic for B-trees: When it 
comes time to split a node because it contains M entries, we combine the node with 
its sibling. If the sibling has k entries with k <M~ 1, we reallocate the items giving the 
sibling and the full node each about (M+k)/2 entries. Otherwise, we create a new node 
and give each of the three nodes about 2M/3 entries. Also, we allow the root to grow 
to hold about 4M/3 items, splitting it and creating a new root node with two entries 
when it reaches that bound. State bounds on the number of probes used for a search 
or an insertion in a lU-tree of order M with N items. Compare your bounds with the 
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corresponding bounds for B-trees (see proposition b). Develop an insert implementa¬ 
tion for B*-trees. 

6.21 Write a program to compute the average number of external pages for a B-tree of 
order Mbuilt from N random insertions into an initially empty tree. Run your program 
for reasonable values of M and N. 

6.22 If your system supports virtual memory, design and conduct experiments to com¬ 
pare the performance of B-trees with that of binary search, for random searches in a 
huge symbol table. 

6.23 For your internal-memory implementation of Pag e in exercise 6 . 15 , run experi¬ 
ments to determine the value of M that leads to the fastest search times for a B-tree 
implementation supporting random search operations in a huge symbol table. Restrict 
your attention to values of M that are multiples of 100. 

6.24 Run experiments to compare search times for internal B-trees (using the value of 
M determined in the previous exercise), linear probing hashing, and red-black trees for 
random search operations in a huge symbol table. 
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EXERCISES on suffix arrays 

6.25 Give, in the style of the figure on page 882, the suffixes, sorted suffixes, i ndex () and 
1 cp() tables for the following strings: 

a. abacadaba 

b. mississippi 

c. abcdefghij 

d. aaaaaaaaaa 

6.26 Identify the problem with the following code fragment to compute all the suffixes 
for suffix sort: 

suffix = 

for (int i = s.lengthO - 1; i >= 0; i--) 

{ 

suffix = s.charAt(i) + suffix; 
suffixes[i] = suffix; 

} 

Answer : It uses quadratic time and quadratic space. 

6.27 Some applications require a sort of cyclic rotations of a text, which all contain all 
the characters of the text. For i from 0 to N— 1, the / th cyclic rotation of a text of length 
N is the last N — i characters followed by the first i characters. Identify the problem with 
the following code fragment to compute all the cyclic rotations: 

int N = s.1ength(); 

for (int i =0; i < N; i++) 

rotation[i] = s.substring(i, N) + s.substring(0, i); 

Answer : It uses quadratic time and quadratic space. 

6.28 Design a linear-time algorithm to compute all the cyclic rotations of a text string. 
Answer: 

String t = s + s; 

int N = s.1ength(); 

for (int i =0; i < N; i++) 

rotation[i] = r.substring(i, i + N); 
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6.29 Under the assumptions described in section 1 . 4 . give the memory usage of a 
Suffi xArray object with a string of length N. 

6.30 Longest common substring. Write a Suffi xArray client LCS that take two file¬ 
names as command-line arguments, reads the two text files, and finds the longest sub¬ 
string that appears in both in linear time. (In 1970, D. Knuth conjectured that this 
task was impossible.) Hint-. Create a suffix array for s#t where s and t are the two text 
strings and # is a character that does not appear in either. 

6.31 Burrows-Wheeler transform. The Burrows-Wheeler transform (BWT) is a trans¬ 
formation that is used in data compression algorithms, including bzi p2 and in high- 
throughput sequencing in genomics. Write a Suffi xArray client that computes the 
BWT in linear time, as follows: Given a string of length N (terminated by a special end- 
of-file character $ that is smaller than any other character), consider the N-by-N matrix 
in which each row contains a different cyclic rotation of the original text string. Sort 
the rows lexicographically. The Burrows-Wheeler transform is the rightmost column 
in the sorted matrix. For example, the BWT of mi ssi ssi ppi $ is i pssmSpi ssi i . The 
Burrows-Wheeler inverse transform (BWI) inverts the BWT. For example, the BWI of 
i pssmSpi ssi i is mi ssi ssi ppi $. Also write a client that, given the BWT of a text string, 
computes the BWI in linear time. 

6.32 Circular string linearization. Write a Suffi xArray client that, given a string, finds 
the cyclic rotation that is the smallest lexicographically in linear time. This problem 
arises in chemical databases for circular molecules, where each molecule is represented 
as a circular string, and a canonical representation (smallest cyclic rotation) is used to 
support search with any rotation as key. (See exercise 6.27 and exercise 6 . 28 .) 

6.33 Longest k-repeated substring. Write a Suffi xArray client that, given a string and 
an integer k, find the longest substring that is repeated k or more times. 

6.34 Long repeated substrings. Write a Suffi xArray client that, given a string and an 
integer L, finds all repeated substrings of length L or more. 

6.35 k-gram frequency counts. Develop and implement an ADT for preprocessing a 
string to support efficiently answering queries of the form How many times does a given 
k-gram appear ? Each query should take time proportional to k log N in the worst case, 
where N is the length of the string. 
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EXERCISES on maxflow 

6.36 If capacities are positive integers less than M, what is the maximum possible flow 
value for any sf-network with V vertices and E edges? Give two answers, depending on 
whether or not parallel edges are allowed. 

6.37 Give an algorithm to solve the maxflow problem for the case that the network 
forms a tree if the sink is removed. 

6.38 True or false. If true provide a short proof, if false give a counterexample: 

a. In any max flow, there is no directed cycle on which every edge carries 
positive flow 

b. There exists a max flow for which there is no directed cycle on which every 
edge carries positive flow 

c. If all edge capacities are distinct, the max flow is unique 

d. If all edge capacities are increased by an additive constant, the min cut 
remains unchanged 

e. If all edge capacities are multiplied by a positive integer, the min cut re¬ 
mains unchanged 

6.39 Complete the proof of proposition g: Show that each time an edge is a critical 
edge, the length of the augmenting path through it must increase by 2. 

6.40 Find a large network online that you can use as a vehicle for testing flow algo¬ 
rithms on realistic data. Possibilities include transportation networks (road, rail, or 
air), communications networks (telephone or computer connections), or distribution 
networks. If capacities are not available, devise a reasonable model to add them. Write 
a program that uses the interface to implement flow networks from your data. If war¬ 
ranted, develop additional private methods to clean up the data. 

6.41 Write a random-network generator for sparse networks with integer capacities 
between 0 and 2 20 . Use a separate class for capacities and develop two implementa¬ 
tions: one that generates uniformly distributed capacities and another that generates 
capacities according to a Gaussian distribution. Implement client programs that gener¬ 
ate random networks for both weight distributions with a well-chosen set of values of 
V and E so that you can use them to run empirical tests on graphs drawn from various 
distributions of edge weights. 
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6.42 Write a program that generates V random points in the plane, then builds a flow 
network with edges (in both directions) connecting all pairs of points within a given 
distance d of each other, setting each edge’s capacity using one of the random models 
described in the previous exercise. 

6.43 Basic reductions. Develop FordFul kerson clients for finding a maxflow in each 
of the following types of flow networks: 

■ Undirected 

■ No constraint on the number of sources or sinks or on edges entering the source 
or leaving the sink 

■ Lower bounds on capacities 

■ Capacity constraints on vertices 

6.44 Product distribution. Suppose that a flow represents products to be transferred by 
trucks between cities, with the flow on edge u-v representing the amount to be taken 
from city u to city v in a given day Write a client that prints out daily orders for truck¬ 
ers, telling them how much and where to pick up and how much and where to drop off. 
Assume that there are no limits on the supply of truckers and that nothing leaves a given 
distribution point until everything has arrived. 

6.45 Job placement. Develop a FordFul kerson client that solves the job-placement 
problem, using the reduction in proposition j. Use a symbol table to convert symbolic 
names into integers for use in the flow network. 

6.46 Construct a family of bipartite matching problems where the average length of 
the augmenting paths used by any augmenting-path algorithm to solve the correspond¬ 
ing maxflow problem is proportional to E. 

6.47 st-connectivity. Develop a FordFul kerson client that, given an undirected graph 
G and vertices s and t, finds the minimum number of edges in G whose removal will 
disconnect t from s. 

6.48 Disjoint paths. Develop a FordFul kerson client that, given an undirected graph 
G and vertices s and t, finds the maximum number of edge-disjoint paths from s to t. 
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EXERCISES on reductions and intractability 

6.49 Find a nontrivial factor of 37703491. 

6.50 Prove that the shortest-paths problem reduces to linear programming. 

6.51 Could there be an algorithm that solves an NP-complete problem in an average 
time of N lo s N , if P * NP? Explain your answer. 

6.52 Suppose that someone discovers an algorithm that is guranteed to solve the bool¬ 
ean satisfiability problem in time proportional to 1.1 N Does this imply that we can solve 
other NP-complete problems in time proportional to 1.1 v ? 

6.53 What would be the significance of a program that could solve the integer linear 
programming problem in time proportional to 1.1 N ? 

6.54 Give a poly-time reduction from vertex cover to 0-1 integer linear inequality 
satisfiability. 

6.55 Prove that the problem of finding a Hamiltonian path in a directed graph is NP- 
complete, using the NP-completeness of the Hamiltonian-path problem for undirected 
graphs. 

6.56 Suppose that two problems are known to be NP-complete. Does this imply that 
there is a poly-time reduction from one to the other? 

6.57 Suppose that X is NP-complete, X poly-time reduces to Y, and Y poly-time reduces 
to X. Is Y necessarily NP-complete? 

Answer : No, since Y may not be in NP. 

6.58 Suppose that we have an algorithm to solve the decision version of boolean satisfi¬ 
ability, which indicates that there exists and assignment of truth values to the variables 
that saitfies the boolean expression. Show how to find the assignment. 

6.59 Suppose that we have an algorithm to solve the decision version of the vertex 
cover problem, which indicates that there exists a vertex cover of a given size. Show how 
to solve the optimization version of finding the vertex cover of minimum cardinality. 

6.60 Explain why the optimization version of the vertex cover problem is not neces¬ 
sarily a search problem. 
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Answer: There does not appear to be an efficient way to certify that a purported solu¬ 
tion is the best possible (even though we could use binary search on the search version 
of the problem to find the best solution). 

6.61 Suppose that X and Y are two search problems an that X poly-time reduces to Y. 
Which of the following can we infer? 

a. If Y is A/P-complete then so is X. 

b. If X is A/P-complete then so is Y. 

c. If X is in P, then Y is in P. 

d. If Y is in P, then X is in P. 

6.62 Suppose that P * NP. Which of the following can we infer? 

e. If X is A/P-complete, then X cannot be solved in polynomial time. 
f If X is in A/P, then X cannot be solved in polynomial time. 

g. If X is in A/P but not A/P-complete, then X can be solved in polynomial time. 

h. If X is in P, then X is not A/P-complete. 
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longest path 912 
maze exploration 530 
path finding 535-537 


reachability 570-573 
strong components 584-591 
topological order 574-581 
transitive closure 592 
Tremaux exploration 530 
2-colorability 547 
union-find 546 
Depth of a node 226 
Deque data type 167,212 
Design by contract 107 
Deterministic finite state au¬ 
tomaton 764 
Devroye, L. 412 
DFA. See Deterministic finite 
state automaton 
Diameter of a graph 559, 685 
Dictionary 361. 
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Dijkstra’s 2-stack algo¬ 
rithm 128-131 
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shortest paths 638-693 
strong components 584-591 
symbol graphs 548 
transitive closure 592-593 
union-find 216-241 
Greatest common divisor 4 
Greedy algorithm 
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Huffman encoding 830 
minimum spanning tree 607 
Grep 804 

H 

Halting problem 910 
Hamiltonian cycle 562, 920 
Hamiltonian path 598,913,920 
Handle 112 
Hard-disc model 856 
Harmonic number 23,185 
Harmonic sum 185 
hashCodeO method 101, 102, 
461-462 

Hash function 458,459-463 
modular 459 
perfect 480 

Rabin-Karp algorithm 774 
Hashing. See Hash function; 

See also Hash table 
hash function 459-463 
time-space tradeoff 458 
Hash table 458-485 
array resizing 474-475 
clustering 472 
collision resolution 458 
cuckoo hashing 484 
deletion 468 
double hashing 483 
double probing 483 
duplicate keys 488 
hashCodeO method 461-462 
hash function 458 
Java library 489 
linear probing 469-474 
load factor 471 
memory usage of 476 
primitive types 488 
separate chaining 464-468 
uniform hashing assump¬ 
tion 463 
Head vertex 566 


Heap. See Binary heap 
multiway 319 
Heap order 313 
Heapsort 323-327 
Height 

2-3 search tree 429 
binary search tree 412 
complete binary tree 314 
red-black BST 444 
tree 226 

Hibbard deletion 422 
Hibbard, T. 410 
Hoare, C. A. R. 205 
Horner’s method 460 
h-sorted array 258 
Huffman compression 350, 
826-838 
analysis of 833 
optimality of 833 
Huffman, D. 827 

I 

if statement 15 
i f - e 1 s e statement 15 
Immutability 105-106 
defensive copy 112 
of strings 114, 202, 696 
priority queue keys 320 
symbol table keys 365 
Implementation 28, 88 
Implementation inheritance 101 
import statement 27,29,66 
Incident edge 519 
Increment sequence 258 
In data type 41,83 
Indegree of a vertex 566 
Index 361,496-501 
a string 877 
files 500-501 
inverted 498-501 
Index priority queue 320-322 
Dijkstra’s algorithm 652 


Prim’s algorithm 620 
Indirect sort 286 
Ineligible edge 

minimum spanning tree 616 
shortest paths 646 
Infix notation 13,128,162 
Inherited methods 66, 100-101 
compareO 338—339 
compareToO 246—247 
equals() 102—103 
getClassO 101 
hashCodeO 101,461—462 
hasNextO 138 
iteratorO 138 
next() 138 
toStringO 66, 101 
Inner loop 180,184,195 
Inorder tree traversal 412 
In-place merge 270 
Input and output 82-83 
binary data 812-815 
from a file 41 
piping 40 
redirection 40 
Input model 197 
Input size 173 
Insertion sort 250-252 
Instance method 65, 84 
Instance variable 84 
i nt primitive data type 12 
Integer linear inequality satisfi¬ 
ability problem 913 
Integer linear programming 920 
Integer overflow 51 
Interface 100 
Comparable 246—247 
Comparator 338—340 
Iterable 138 
Iterator 139 
Interface inheritance 100 
Interior point method 909 
Internal path length 412 
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Internet DNS 493 
Internet Movie Database 497 
Interpreter 130 
Interval graph 564 
Intractability 910-921 
Inversion 252, 286 
Inverted index 498-501 
Ising model 920 
Isomorphic graph 561 
Item 

contains a key 244 
sorting 244 
symbol table 387 
with multiple keys 339 
Item type parameter 134 
Iteration 123, 138-141 
fail-fast 171 
foreach loop 123 

J 

Jacquet, P. 882 
Jarnik’s algorithm 628. 

See also Prim’s algorithm 
Jarnik,V. 628 
Java programming 
array 18-21 
arrays as objects 72 
arrays of objects 72 
assertion 107 
assert statement 107 
assignment statement 14 
autoboxing 122 
autounboxing 122 
base class 101 
bitwise operators 52 
block statement 15 
boolean expression 13 
break statement 15 
bytecode 10 
cast 13 
class 10, 64 
classpath 66 


comparison operator 13 
conditional statement 15 
constructor 65, 84 
conti nue statement 15 
covariant arrays 158 
create an object 67 
declaration statement 14 
default initialization 18, 86 
deprecated method 113 
derived class 101 
Error 107 
Exception 107 
expression 11, 13 
final modifier 84, 105-106 
for loop 16 
foreach loop 123 
garbage collection 104 
generic array creation 158 
generics 122-123,134—135 
identifier 11 
if statement 15 
i f-el se statement 15 
implicit assignment 16 
import statement 29 
imported system libraries 27 
infix expression 13 
inheritance 100-101 
inherited method 66 
initializing declarations 16 
inner class 159 
instance method 65, 84, 86 
instance method signature 86 
instance variable 84 
invoke instance method 68 
iterable collections 123 
just-in-time compiler 195 
literal 11 
loitering 137 
loop statement 15 
memory management 104 
modular programming 26 
nested class 159 


new() 67 
objects 67-74 
objects as arguments 71 
objects as return values 71 
operator 11 
operator precedence 13 
orphan 137 
orphaned object 104 
overloading 12, 24 
override a method 101 
parameterized type 122,134 
pass by reference 71 
pass by value 24,71 
primitive data type 11-12 
private class 159 
private modifier 84 
protected modifier 110 
public modifier 84,110 
ragged array 19 
recursion 25 
reference 67 
reference type 64 
return statement 86 
scope 14, 87 
short-circuit operator 52 
side effects 24 
single-statement blocks 16 
standard libraries 27 
standard system libraries 27 
statement 14 
static method 22-25 
static variable 113 
strong typing 14 
subclass 101 
superclass 101 
this reference 87 
throw an error/exception 107 
two-dimensional array 19 
type conversion 13, 35 
type erasure 158 
type parameter 122 
unit testing 26 
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using objects 69 
variable 11 
visibility modifier 84 
while loop 15 
wrapper type 122 
Java system sort 306 
Java virtual machine 51 
java.awt 
Color 75 
Font 75 
j ava.io 
File 75 
java.lang 

ArithmeticException 107 

ArraylndexOutOfBounds 107 

Boolean 102 

Byte 102 

Character 102 

ClassCastException 387 

Comparable 100 

Double 34, 102 

Float 102 

Integer 102 

Iterable 100, 123, 138, 154 
Long 102 
Math 28 

NullPointer 107, 113, 159 
Object 101 

OutOfMemoryError 107 
RuntimeException 107 
Short 102 

StackOverflowError 57, 107 
StringBuiIder 27,105,697 
UnsupportedOperation 139 
j ava.net 
URL 75 
java.uti1 
ArrayList 160 
Arrays 29 

Comparator 100, 339 
ConcurrentModification 160 
Date 113 


HashMap 489 

Iterator 100, 138-141, 154 
LinkedList 160 
NoSuchElementException 139 
PriorityQueue 352 
Stack 159 
TreeMap 489 
Job-scheduling problem. 

See Scheduling 
Josephus problem 168 
Just-in-time compiler 195 

K 

Karp, R. 901 
Karp, R. M. 759 
Kendall tau distance 286, 345, 
356 

Kevin Bacon number 553-554 
Key 244 
Key equality 

ordered symbol table 368 
symbol table 365 
Key-indexed counting 703-705 
Key type parameter 
priority queue 309 
symbol table 361 
Keyword in context 879 
Khachian, L. G. 909 
Kleene’stheorem 794 
Knuth, D. E. 178,205,759 
Knuth-Morris-Pratt 762-769 
Knuth shuffle 32 
Kosaraju’s algorithm 586-590 
Kruskal, J. 628 
Kruskal’s algorithm 624-627 
KWIC. See Keyword-in-context 

L 

Last-in-hrst-out policy 127 
Las Vegas algorithm 778 
Leading-term approximation. 


See Tilde notation 
Least-significant digit. 

See LSD string sort 
Leipzig Corpora Collection 371 
Lempel,A. 839 
lessQ method 245,315 
Level-order traversal 
binary heap 313 
binary search tree 420 
Levin, L. 918 

LIFO. See Last-in first-out policy 
LIFO stack. See Stack data type 
Linear equation satisfiability 913 
Linear inequality satisfiabil¬ 
ity 913 

Linear probing 469-474 
Linear programming 907-909 
ellipsoid algorithm 909 
interior point method 909 
reductions 907-909 
simplex algorithm 909 
Linear running time 186 
Linearithmic running time 186 
Linked allocation 156 
Linked list 142-146 
building 143 
circular 165 
defined 142 
deletion 145 

deletion from beginning 145 
garbage collection 145 
insertion 145 
insertion at beginning 144 
insertion at end 145 
iterator 154-155 
memory usage of 201 
Node data type 142 
queue 150 
reverse a 165 
sequential search 374 
shuffle a 286 
sort a 286 
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stack 147-149 
traversal 146 
Literal 

null 112-113 
primitive type 11 
string 80 

Load-balancing 349,909 
Load factor 471 
Local minimum 210 
Logarithm function 
binary 185 
integer binary 185 
natural 185 

Logarithmic running time 186 
Log-log plot 176 
Loitering 137 

Longest common prefix 875 
Longest paths 661,911 
Longest prefix match 842 
Longest-processing-time first 
rule 349 

Longest repeated substring 875 
1 ong primitive data type 13 
Loop 
for 16 
foreach 138 
inner 180 
while 15 

Lossless data compression 811 
Lossy data compression 811 
Lower bound 
priority queue 332 
sorting 279-282 
3-sum problem 190 
union-find 231 
Lowest common ancestor 598 
Loyd, S. 358 
LSD string sort 706-709 
LZW algorithm 839-845 
compression 840 
expansion 841 
trie representation 840 


M 

Manber, U. 884 

Mark-and-sweep garbage collec¬ 
tion 573 
Maslow,A. 904 
Maslow’s hammer 904 
Matrix data type 60 
Maxflow-mincut theorem 894 
Maxflow problem 886-902. 

See also Mincut problem 
Ford-Fulkerson 891-893 
integrality property 894 
maxflow-mincut theo¬ 
rem 892-894 

max bipartite matching 906 
preflow-push algorithm 902 
reductions 905-907 
residual network 895-897 
Maximum 
in array 30 
in binary heap 313 
in binary search tree 406 
in ordered symbol table 367 
Maximum st-flow problem. 

See Maxflow problem 
Max bipartite matching 906 
Maze 530 

Mcllroy, D. 298, 306 
McKellar, A. 306 
Median 332,345-347 
Median-of-3 partitioning 305 
Memory management 104 
linked allocation 156 
loitering 137 
orphan 137 

Sequential allocation 156 
Memory usage 200-204 
array 202 
hash table 476 
linked list 201 
nested class 201 
object 67, 201 


primitive types 200 
R-waytrie 744 
stack 213 
string 202 
substring 202-204 
Mergesort 270-288 
abstract in-place merge 270 
analysis of 272 
bottom-up 277 
linked list 279, 286 
multiway 287 
natural 285 
optimality 282 
stability 341 
top-down 272 
Merging 270-271 
Method 

inherited 100-101 
instance 68-69, 86-87 
static 22-25 

Mincut problem 893. See 

also Maxflow problem 
Minimum 
in array 30 

in binary search tree 406 
in ordered symbol table 367 
Min spanning forest 605 
Min spanning tree 604-637 
Boruvka’s algorithm 636 
bottleneck shortest paths 690 
critical edge 633 
crossing edge 606 
cut 606 

cut optimality conditions 634 
cut property 606 
defined 604 
greedy algorithm 607 
Kruskal’s algorithm 624-627 
Prim’s algorithm 616-623 
reverse-delete algorithm 633 
Vyssotsky’s algorithm 633 
Minimum sf-cut problem. 


www.it-ebooks.info 


944 


INDEX 


See Mincut problem 
Minotaur 530 

Mismatched character rule 770 
M. L. Fredman 628 
Modular hash function 459, 774 
Modular programming 26 
Monte Carlo algorithm 776 
Moore, J. S. 759 
Moore’s law 194-195 
Morris, J. H. 759 
Most-significant-digit sort. 

See MSD string sort 
Move-to-front 169 
MSD string sort 710-718 
Multidimensional sort 356 
Multigraph 518 
Multiple-source reachability 
problem 570, 797 
Multiset 509 
Multiway mergesort 287 
Multiway trie. See R-way trie 
Myers, E. 884 

N 

Natural logarithm function 185 
Natural mergesort 285 
Natural order 337 
Negative cost cycle. 

See Negative cycle 
Negative cycle 668-670, 

677-681 

Nested class 159 
Network flow. 

See Maxflow problem 
new() 67 

Newton’s method 23 
NFA. See Nondeterministic 
finite-state automata 
Node data type 159 
bag 155 

binary search tree 398 
Huffman trie 828 


linked list 142 
queue 151 
red-black BST 433 
R-way trie 734 
stack 149 

ternary search trie 747 
Nondeterminism 794 
Turing machine 914 
Nondeterministic finite-state 
automata 794—799 
A/P 912 

/VP-complete 917-918 

Null link 396 

null literal 112-113 

o 

Object 67-74. See also Object- 
oriented programming 
behavior 67, 73 
identity 67, 73 
memory usage of 201 
state 67, 73 

Object-oriented program¬ 
ming 64-119 
arrays of objects 72 
creating an object 67 
declaring an object 67 
encapsulation 96 
inheritance 100 
instance 73 

instantiate an object 67 
invoke instance method 68 
objects 67-74 
objects as arguments 71 
objects as return values 71 
reference 67 
subtyping 100 
using objects 69 
Odd-length cycle in a graph 562 
OOP. See Object-oriented pro¬ 
gramming 

Operations research 349 


Optimization problem 913 
Ordered symbol table 366-369 
floor and ceiling 367 
minimum and maximum 367 
ordered array 378 
range query 368 
rank and selection 367 
red-black BST 446 
Order of growth 179 
Order-of-growth classifica¬ 
tions 186-188 

Order-of-growth hypothesis 180 
Order statistic 345 
binary search tree 406 
ordered symbol table 367 
quickselect 345-347 
Orphaned object 104,137 
Out data type 41, 83 
Outdegree of a vertex 566 
Output. See Input and output 
Overflow 51 
Overloading 
constructor 84 
static method 24 
Overriding a method 66,101 

P 

P complexity class 914 
P= A/P question 916 
Page data type 870 
Palindrome 81, 783 
Parallel arrays 
linear probing 471 
ordered symbol table 378 
sorting 357 

Parallel edge 518,566,612,640 
Parallel job scheduling 663-667 
Parallel precedence-constrained 
scheduling 663, 904 
Parameterized type. See Generics 
Parent-link representation 
breadth-first search tree 539 
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depth-first search tree 535 
minimum spanning tree 620 
shortest-paths tree 640 
union-find 225 
Parsing 

an arithmetic expression 128 
a regular expression 800-804 
Particle data type 860 
Partitioning algorithm 290 

2- way 288 

3- way (Bentley-Mcllroy) 306 
3-way (Dijkstra) 298 
median-of-3 296, 305 
median-of-5 305 
selection 346-347 

Partitioning item 290 
Pass by reference 71 
Pass by value 24,71 
Path. See Longest paths; 

See also Shortest paths 
augmenting 891 
Hamiltonian 913, 920 
in a digraph 567 
in a graph 519 
length of 519,567 
simple 519,567 
Path compression 231 
Pattern matching. 

See Regular expression 
Perfect hash function 480 
Performance. See Propositions 
Permutation 
Kendall-tau distance 356 
random 168 
ranking 345 
sorting 354 
Phone book 492 
Picture data type 814 
Piping 40 
Point data type 77 
Pointer 111. See also Reference 
safe 112 


Pointer sort 338 
Poisson approximation 466 
Poisson distribution 466 
Polar angle 356 
Polar coordinate 77 
Polar sort 356 
Poly-time reduction 916 
Pop operation 127 
Postfix notation 162 
Postorder traversal 
of a digraph 578 
reverse 578 
Power law 178 
Pratt, V.R. 759 

Precedence-constrainted sched¬ 
uling 574-575 
Precedence order 
arithmetic expressions 13 
regular expressions 789 
Prefix-free code 826-827 
compression 829 
expansion 828 
Huffman 833 
optimal 833 

reading and writing 834-835 
trie representation 827 
Preorder traversal 
of a digraph 578 
of a trie 834 

Prime number 23, 774, 785 
Primitive data type 11-12 
memory usage of 200 
reason for 51 
wrapper type 102 
Primitive type 
versus reference type 110 
Prim, R. 628 

Prim’s algorithm 350,616-623 
eager 620-623 
lazy 616-619 
Priority queue 308-335 
binary heap 313-322 


change priority 321 
delete 321 

Dijkstra’s algorithm 652 
Fibonacci heap 628 
Huffman compression 830 
index priority queue 320-321 
linked-list 312 
multiway heap 319 
ordered array 312 
Prim's algorithm 616 
reductions 345 
remove the minimum 321 
soft heap 629 
stability 356 
unordered array 310 
private access modifier 84 
Probabilistic algorithm. See Ran¬ 
domized algorithm 
Probe 471 
Problem size 173 
Programs 
Accumulator 93 
AcyclicLP 661 
AcyclicSP 660 
Arbitrage 680 
Average 39 
Bag 155 

BellmanFordSP 674 
BinaryDump 814 
BinarySearch 47 
Bi narySearchST 379,381,382 
BlackFilter 491 
BoyerMoore 772 
BreadthFirstPaths 540 
BST 398,399,407,409,411 
BTreeSET 872 
Cat 82 
CC 544 

CollisionSystem 863—864 
Count 699 
Counter 89 
CPM 665 
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Cycle 547 
Date 91, 103, 247 
DeDup 490 

DegreesOfSeparation 555 
DepthFirstOrder 580 
DepthFirstPaths 536 
DepthFirstSearch 531 
Digraph 569 
DijkstraAllPairsSP 656 
DijkstraSP 655 
DirectedCycle 577 
DirectedDFS 571 
DirectedEdge 642 
DoublingTest 177 
Edge 610 

EdgeWeightedDigraph 643 
EdgeWeightedGraph 611 
Evaluate 129 
Event 861 
Example 245 
Filelndex 501 
FixedCapacityStack 135 
FixedCapacityStackOf- 
Strings 133 
Flips 70 
FlipsMax 71 
FlowEdge 896 
FordFulkerson 898 
FrequencyCounter 372 
Genome 819—820 
Graph 526 
GREP 804 
Heap 324 
HexDump 814 
Huffman 836 
Insertion 251 
KMP 768 

KosarajuSCC 587 
KruskalMST 627 
KWIC 881 
LazyPrimMST 619 
LinearProbingHashST 470 
LookupCSV 495 


Lookuplndex 499 

LRS 880 

LSD 707 

LZW 842,844 

MaxPQ 318 

Merge 271, 273 

MergeBU 278 

MSD 712 

Multiway 322 

NFA 799,802 

PictureDump 814 

PrimMST 622 

Queue 151 

Quick 289,291 

Quick3string 720 

Quick3way 299 

RabinKarp 777 

RedBlackBST 439 

ResizingArrayQueue 140 

ResizingArrayStack 141 

Reverse 127 

RLE 824 

Rolls 72 

Selection 249 

SeparateChai ningHashST 465 

SequentialSearchST 375 

SET 489 

Shell 259 

SortCompare 256 

SparseVector 503 

Stack 149 

StaticSEToflnts 99 

Stats 125 

Stopwatch 175 

SuffixArray 883 

SymbolGraph 552 

ThreeSum 173 

ThreeSumFast 190 

TopM 311 

Topological 581 

Transaction 340 

TransitiveClosure 593 

TrieST 737—741 


TST 747 
TwoColor 547 
TwoSumFast 189 
UF 221 

VisualAccumulator 95 
WeightedQuickUnionUF 228 
WhiteFilter 491 
Whitelist 99 
Properties 180 
3-sum 180 

Boyer-Moore algorithm 773 
insertion sort 255 
quicksort 343 
Rabin-Karp algorithm 778 
red-black BST 445 
selection sort 255 
separate-chaining 467 
shellsort 262 
versus proposition 183 
Propositions 182 

2- 3 search tree 429 

3- sum 182 

3-way quicksort 301 
3-way string quicksort 723 
arbitrage 681 
B-tree 871 

Bellman-Ford 671, 673 
binary heap 319 
binary search 383 
BST 403-404,412 
breadth-first search 541 
brute substring search 761 
complete binary tree 314 
connected components 546 
Cook-Levin theorem 918 
critical path method 666 
cut property 606 
DFS 531,537,570 
Dijkstra’s algorithm 652, 654 
flow conservation 893 
Ford-Fulkerson 900-901 
generic shortest-paths 651 
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greedy MST algorithm 607 
heapsort 323, 326 
Huffman algorithm 833 
index priority queue 321 
insertion sort 250, 252 
integer programming 917 
key-indexed counting 705 
Knuth-Morris-Pratt 769 
Kosaraju’s algorithm 588, 590 
Kruskal’s algorithm 624, 625 
linear-probing hash table 475 
linear programming 908 
longest paths in DAG 661 
longest repeated substring 885 
LSD string sort 706, 709 
maxflow-mincut theorem 894 
maxflow reductions 906 
mergesort 272, 279, 282 
MSD string sort 717,718 
negative cycles 669 
parallel job scheduling with 
relative deadlines 667 
particle collision 865 
Prim’s algorithm 616,618,623 
quick-find algorithm 223 
quickselect 347 
quicksort 293-295 
quick-union algorithm 226 
red-black BST 444, 447 
regular expression 799, 804 
resizing-array stack 199 
R-way trie 742, 743, 744 
selection sort 248 
separate-chaining 466, 475 
sequential search 376 
shortest paths in DAG 658 
shortest-paths optimality 650 
shortest paths reductions 905 
sorting lower bound 280, 300 
sorting reductions 903 
suffix array 882 
ternary search trie 749, 751 


topological order 578, 582 
universal compression 816 
weighted quick-union 229 
protected modifier 110 
Protein folding 920 
public access modifier 110 
Pushdown stack 127. 

See also Stack data type 
Push operation 127 


Q 


Quadratic running time 186 
Quantum computer 911 
Queue data type 
analysis of 198 
API 126 

circular linked list 165 
linked-list 150-151 
resizing-array 140 
Quick-find algorithm 222-223 
Quickselect 345-347 
Quicksort 288-307 

2- way partitioning 290 

3- way partitioning 298-301 
3-way string 719 
analysis of 293-295 

and binary search trees 403 
duplicate keys 292 
function-call stack size 304 
median-of-2 296, 305 
median-of-5 305 
nonrecursive 306 
random shuffle 292 
Quick-union 224-227 
path compression 231 
weighted 227-230 


R 

Rabin-Karp algorithm 774-778 
Rabin, M. O. 759 
Radius of a graph 559 


Radix 700 

Radix sorting. See String sorting 
Random bag data type 167 
Randomized algorithm 198 
Las Vegas 778 
Monte Carlo 776 
quickselect 345-347 
quicksort 290, 307 
Rabin-Karp algorithm 776 
3-way string quicksort 722 
Random number 30-32 
Random queue data type 168 
Random string model 716-717 
Range query 
binary search tree 412 
ordered symbol table 368 
Rank 

binary search 25,378-381 
binary search tree 408,415 
ordered symbol table 367 
suffix array 879 
Reachability 570-572, 590 
Reachable vertex 567 
Recurrence relation 
binary search 383 
mergesort 272 
quicksort 293 
Recursion 25. 

See also Base case; 

See also Recursion 
binary search 25, 380 
binary search tree 401 
depth-first search 531 
Euclid’s algorithm 4 
Fibonacci numbers 57 
mergesort 272 
quicksort 289 
Red-black BST 432-447 
and 2-3 search tree 432 
analysis of 444-447 
color flip 436 
color representation 433 
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defined 432 

delete the maximum 454 
delete the minimum 453 
deletion 441-443,455 
implementation 439 
insertion 437-439 
left-leaning 432 
perfect black balance 432 
rotation 433-434 
search 432 
Redirection 40 
Reduction 903-909 
defined 903 
polynomial-time 916 
linear programming 907-909 
maxflow 905-907 
priority queue 345 
shortest-paths 904-905 
sorting 344—347, 903-904 
Reference 67 
Reference type 64 
Reflexive relation 102, 216, 247, 
584 

Regular expression 82, 788 
building an NFA 800-804 
closure operation 789 
concatenation operation 789 
defined 790 
epsilon-transition 795 
match transition 795 
nondeterministic finite-state 
automaton 794-799 
or operation 789 
parentheses 789 
\\s+ 82 
shortcuts 791 

simulating an NFA 797-799 
Rehashing 474 
Relation 

antisymmetric 247 
equivalence 102,216,584 
reflexive 102, 216, 247, 584 


symmetric 102, 216, 584 
total order 247 
transitive 102, 216, 247, 584 
Residual network 895-897 
Resizing array 136-137 
binary heap 320 
hash table 474-475 
queue 140 
stack 136 
Return value 22 
Reverese postorder traversal 578 
Reverse 

a linked list 165-166 
an array 21 
array iterator 139 
with a stack 127 
Reverse-delete algorithm 633 
Reverse graph 586 
Reverse Polish notation. 

See Postfix notation 
Reverse postorder 578 
Ring buffer data type 169 
RLE. See Run-length encoding 
Robson, J. 412 
Rooted tree 640 
Rotation in a BST 433-434, 452 
Run-length encoding 822-825 
Running time 172-173 
analysis of 176 
constant 186 
cubic 186 
doubling ratio 192 
exponential 186 
inner loop 180 
linear 186 
logarithmic 186 
measuring 174 
order of growth 179 
quadratic 186 

tilde approximation 178-179 
Run-time error.See Error; See 
also Except! on 


R-waytrie 730-744 
Alphabet 741 
analysis of 742-743 
collecting keys 738 
deletion 740 
insertion 734 
longest prefix 739 
memory usage of 744 
one-way branching 744—745, 
representation 734 
search 732-733 
wildcard match 739 

s 

Safe pointer 112 

Sample mean 30 

Samplesort 306 

Sample standard deviation 30 

Sample variance 30 

Scheduling 

critical-path method 664-666 
load-balancing problem 349 
LPT first 349 
parallel precedence-con¬ 
strained 663-667 
precedence constraint 574- 
575 

relative deadlines 666 
SPT first 349 
Scientific method 172 
Scope of a variable 14, 87 
Search hit 376 
Searching 360-513. See 
also Symbol table 
Search miss 376 
Search problem 912 
Sedgewick, R. 298 
Selection 345 
binary search tree 406 
ordered symbol table 367 
quickselect 346-347 
suffix array 879 
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Selection client 249 
Selection sort 248-249 
Self-loop 518,566,612,640 
Separate-chaining 464—468 
Sequential allocation 156 
Sequential search 374-377 
Set data type 489-491 
Shannon entropy 300-301 
Shellsort 258-262 
Shortest ancestral path 598 
Shortest augmenting path 897 
Shortest path 638 
Shortest paths problem 638-693 
all-pairs 656 

arbitrage detection 679-681 
Bellman-Ford 668-678 
bitonic 689 
bottleneck 690 
certification 651 
critical edge 690 
Dijkstra’s algorithm 652-657 
edge relaxation 646-647 
edge-weighted DAG 658-667 
generic algorithm 651 
ineligible edge 646 
in Euclidean graphs 656 
monotonic 689 
negative cycle 669 
Negative cycle detection 670 
negative weights 668-681 
optimality conditions 650 
parent-link 640 
reduction 904-905 
shortest-paths tree 640 
single-source 639, 654 
source-sink 656 
undirected graph 654 
vertex relaxation 648 
Shortest-processing-time-first 
rule 349,355 

short primitive data type 13 
Shuffling 


a linked list 286 
an array 32 
quicksort 292 
Side effect 22,108 
Signature 

instance method 86 
static method 22 
Simple digraph 567 
Simple graph 518 
Simplex algorithm 909 
Single-source problems 
connectivity 556 
directed paths 573 
longest paths in DAG 661 
paths 534 
reachability 570 
shortest directed paths 573 
shortest paths in undirected 
graphs 654, 904 
shortest paths 538, 639 
Social network 517 
Soft heap 629 

Software cache 391,451,462 
Sollin, M. 628 
Sorting 242-359. 

See also String sorting 
3-way quicksort 298-301 
binary search tree 412 
certification 246, 265 
Comparable 246—247 
compare-based 279 
complexity of 279-282 
cost model 246 
entropy-optimal 296-301 
extra memory 246 
heapsort 323-327 
indirect 286 
in-place 246 
insertion sort 250-252 
inversion 252 
lower bound 279-282, 306 
mergesort 270-288 


partially-sorted array 252 
pointer 338 
primitive types 343 
quicksort 288-307 
reduction 903-904 
reductions 344-347 
selection sort 248-250 
shellsort 258-262 
stability 341 
suffix array 875-885 
system sort 343 
Source-sink shortest paths 656 
Spanning forest 520 
Spanning tree 520, 604 
Sparse graph 520 
Sparse matrix 510 
Sparse vector 502-505 
Specification problem 97 
SPT. See Shortest paths tree; 

See also Shortest-process- 
ing-time-first rule 
st-cut 892 
sf-flow 888 
sf-flow network 888 
Stability 341, 355 
insertion sort 341 
key-indexed counting 705 
LSD string sort 706 
mergesort 341 
priority queue 356 
Stack data type 127 
analysis of 198,199 
array implementation 132 
fixed-capacity 132-133 
generic 134 
iteration 138-140 
linked-list 147-149 
resizing array 136 
Standard deviation 30 
Standard drawing 36,42-45 
Standard input 36, 39 
Standard libraries 30 
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Draw 82—83 
In 41,82-83 
Out 41, 82—83 
StdDraw 43 
Stdln 39 
StdOut 37 
StdRandom 30 
StdStats 30 
Stopwatch 174—175 
Standard output 36, 37-38 
Static method 22-25 
argument 22 
defining a 22 
invoking a 22 
overloaded 24 
pass by value 24 
recursive 25 
return statement 24 
return value 22 
side effect 22, 24 
signature 22 
Static variable 113 
Statistics 
chi-square 483 
median 345 

minimum and maximum 30 
order 345 

sample mean 30, 125 
sample standard deviation 30 
sample variance 30,125 
StdDraw library 43 
Stdln library 39 
StdOut library 37 
StdRandom library 30 
StdStats library 30 
Steque data type 167,212 
Stirling's approximation 185 
Stopwatch data type 174—175 
String data type 34, 80-81 
API 80 
characters 696 
charAtO method 696 


concatenation 34, 697 
conversion 102 
immutability 696 
indexing 696 
i ndexOf () method 779 
length 696 
lengthQ method 696 
literal 34 

memory usage of 202 
+ operator 80, 697 
substring extraction 696 
substringO method 696 
String processing 80-81, 
694-851 

data compression 810-851 
regular expression 788 
sorting 702-729 
substring search 758-785 
suffix array 875-885 
tries 730-757 
String search. See Substring 
search; See also Trie 
String sorting 702-729 
3-way quicksort 719-723 
key-indexed counting 703 
LSD string sort 706-709 
MSD string sort 710-718 
Strong component 584 
Strong connectivity 584—591 
Strongly connected component. 

See Strong component 
Strongly connected relation 584 
Strongly typed language 14 
Subclass 101 
Subgraph 519 

Sublinear running time 716, 779 
Substring extraction 
memory usage of 202-204 
substringO method 696 
Substring search 758-785 
Boyer-Moore 770-773 
brute-force 760-761 


i ndexOf () method 779 
Knuth-Morris-Pratt 762-769 
Rabin-Karp 774-778 
Sub typing 100 
Suffix array 875-885 
Suffix array data type 879 
Suffix-free code 847 
Superclass 101 
Symbol digraph 581 
Symbol graph 548-555 
Symbol table 360-513 
2-3 search tree 424-431 
API 363,366 
associative array 363 
balanced search tree 424-457 
binary search 378-384 
binary search tree 396-423 
B-tree 866-874 
cost model 369 
defined 362 
duplicate key policy 363 
floor and ceiling 367 
hash table 458-485 
insertion 362 
key equality 365 
lazy deletion 364 
linear-probing 469-474 
minimum and maximum 367 
null value 364 
ordered 366-369 
ordered array 378 
range query 368 
rank and selection 367 
red-black BST 432-447 
R-waytrie 732-745 
search 362 

separate-chaining 464-468 
sequential search 374 
string keys 730-757 
ternary search trie 746-751 
trie 730-757 
unordered linked list 374 
Symmetric order 396 
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Symmetric relation 102, 216, 
584 

Szpankowski, W. 882 

T 

Tail vertex 566 
Tale of Two Cities 371 
Tandem repeat 784 
Tarjan, R. E. 590, 628 
Terminal window 10,36 
Ternary search trie 746-751 
alphabet 750 
analysis of 749 
collecting keys 750 
deletion 750 
insertion 746 

one-way branching 751,755 
prefix match 750 
search 746 
wildcard match 750 
Theseus 530 
this reference 87 
Threading 420 
Tilde notation 178,206 
Time-driven simulation 856 
Timing a program 174—175 
Top-down 2-3-4 tree 441 
Top-down mergesort 272 
Topological sort 574-583 
depth-first search 578 
queue-based algorithm 599 
toStringO method 66, 102 
Total order 247 
Transaction data type 78-79 
compare () 340 
compareToO 266, 337 
hashCodeO 462 
Transitive closure 592 
Transitive relation 102,216, 
247, 584 

Transpose a matrix 56 


Tree. 

2-3 search tree. See 2-3 search 
tree 

binary. See Binary tree 
binary search tree. See Binary 
search tree 

balanced search tree. See Bal¬ 
anced search tree 
binomial 237 
depth of a node 226 
height of 226 
inorder traversal 412 
min spanning tree. See Mini¬ 
mum spanning tree 
parent-link 535, 539 
preorder traversal 834 
rooted 640 
size 226 

spanning tree. See Spanning 
tree 

undirected graph 520 
union-find 224-226 
Tremaux exploration 530 
Triangular sum 185 
Trie 730-757. See also R-way 
trie; See also Ternary 
search trie 
collecting keys 731 
Lempel-Ziv-Welch 840 
longest prefix match 731, 842 
one-way branching 744-745, 
751, 755 

prefix-free code 827 
preorder traversal 834 
reading and writing 834-835 
wildcard match 731 
Tufteplot 456 
Tukeyninther 306 
Turing, A. 910 
Turing machine 910 
Church-Turing thesis 910 
computability 910 


nondeterministic 914 
universality 910 
Type conversion 13 
Type erasure 158 
Type parameter 122,134 

u 

Undecidability 97, 817 
Undirected graph 
acyclic 520 
adjacency-lists 524 
adjacency-matrix 524 
adjacency-sets 527 
adjacent vertex 519 
articulation point 562 
biconnected 562 
bipartite 521, 546-547,562 
breadth-first search 538-542 
bridge 562 
center 559 
connected 519 
connected component 519 
connected to relation 519 
connectivity 534, 543-546 
cycle 519 

cycle detection 546-547 
defined 518 
degree 519 
dense 520 

depth-first search 530-533 
diameter 559 
edge 518 

edge-connected 562 
edge-weighted. 

See Edge-weighted graph 
Euler tour 562 
forest 520 
girth 559 

Elamilton tour 562 
interval graph 564 
isomorphism 561 
multigraph 518 
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odd cycle detection 562 
parallel edge 518 
path 519 
radius 559 
self-loop 518 
simple 518 
simple cycle 519, 567 
simple path 519 
single-source connectivity 556 
single-source paths 534 
single-source shortest 
paths 538 
spanning forest 520 
spanning tree 520 
sparse 520 
subgraph 519 
tree 520 

two-colorability 546-547, 562 

vertex 518 

weighted. 

See Edge-weighted graph 
Unicode 696 
Uniform hashing 463 
Union-find 216-241 
and depth-first search 546 
binomial tree 237 
Boruvka’s algorithm 636 
dynamic connectivity 216 
forest-of-trees 225 
Kruskal's algorithm 625 
parent-link 225 
path compression 231,237 
quick-find 222-223 
quick-union 224—227 
weighted quick-find 236 
weighted quick-union 227- 
231 

weighted quick-union by 
height 237 

weighted quick-union with 
path compression 237 
Uniquely decodable code 826 


Unit testing 26 

Universal data compression 816 

Universality 910 

Upper bound 206, 207, 281 

V 

Val ue type parameter 
symbol table 361 
trie 730 
Variable 10 

Variable-length code 826 
Variance 30 
Vector data type 106 
Vertex 

adjacent 519 
connected to relation 519 
degree of 519 
eccentricity 559 
head and tail 566 
indegree and outdegree 566 
reachable 567 
source 528 

Vertex cover problem 920 
Vertex relaxation 648 
Virtual terminal 10 
Vyssotsky’s algorithm 633 

w 

Web search 496 
Weighted digraph. See Edge- 
weighted digraph 
Weighted edge 604, 638 
Weighted external path 
length 832 

Weighted graph. See Edge- 
weighted graph 

Weighted quick-union 227-231 
Weighted quick-union with path 
compression 237 
Weiner, P. 884 
Welch, T. 839 


while loop 15 

Whitelist filter 8,48-49,99,491 
Wide interface 160, 557 
Wildcard character 791 
Wildcard match 750 
Worst-case guarantee 197 
Wrapper type 102, 122 

z 

Zev, J. 839 

Zero-based indexing 53 
Zipf’s law 393 
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ALGORITHMS 


Fundamentals 

1.1 Pushdown stack (resizing array) 

1.2 Pushdown stack (linked-list) 

1.3 FIFO queue 

1.4 Bag 

1.5 Union-find 

Sorting 

2.1 Selection sort 

2.2 Insertion sort 

2.3 Shellsort 

2.4 Top-down mergesort 
Bottom-up mergesort 

2.5 Quicksort 

Quicksort with 3-way partitioning 

2.6 Heap priority queue 

2.7 Heap sort 

Symbol Tables 

3.1 Sequential search 

3.2 Binary search 

3.3 Binary tree search 

3.4 Red-black BST search 

3.5 Hashing with separate chaining 

3.6 Hashing with linear probing 


Graphs 

4.1 Depth-first search 

4.2 Breadth-first search 

4.3 Connected components 

4.4 Reachability 

4.5 Topological sort 

4.6 Strong componenets (Kosaraju) 

4.7 Minimum spanning tree (Prim) 

4.8 Minimum spanning tree (Kruskal) 

4.9 Shortest paths (Dijkstra) 

4.10 Shortest paths in DAGs 

4.11 Shortest paths (Bellman-Ford) 

Strings 

5.1 LSD string sort 

5.2 MSD string sort 

5.3 Three-way string quicksort 

5.4 Trie symbol table 

5.5 TST symbol table 

5.6 Substring search (Knuth-Morris-Pratt) 

5.7 Substring search (Boyer-Moore) 

5.8 Substring search (Rabin-Karp) 

5.9 Regular expression pattern matching 

5.10 Huffman compression/expansion 

5.11 LZW compression/expansion 
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CLIENTS 


Fundamentals 

Whitelisting 
Expression evaluation 
Connectivity 

Sorting 

Comparing two algorithms 
Top M 

Multiway merge 

Symbol Tables 

Dedup 

Frequency count 
Dictionary lookup 
Index lookup 
File indexing 

Sparse vector with dot product 

Graphs 

Symbol graph data type 
Degrees of separation 
PERT 
Arbitrage 


Strings 

Regular expression pattern matching 
Huffman compression 
Eempel-Ziv-Welch compression 

Context 

Colliding particle simulation 
B-tree set 

Suffix array (elementary) 

Eongest repeated substring 
Keyword in context 
Maxflow (Ford-Fulkerson) 
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